2022-10-18 16:58:12 +02:00
|
|
|
package component
|
|
|
|
|
|
|
|
import (
|
2022-10-31 09:18:36 +01:00
|
|
|
"fmt"
|
2022-10-18 16:58:12 +02:00
|
|
|
"sort"
|
2022-10-20 10:40:46 +02:00
|
|
|
"time"
|
2022-10-18 16:58:12 +02:00
|
|
|
|
2024-09-17 07:33:26 +02:00
|
|
|
"go-mod.ewintr.nl/gte/internal/process"
|
|
|
|
"go-mod.ewintr.nl/gte/internal/storage"
|
|
|
|
"go-mod.ewintr.nl/gte/internal/task"
|
|
|
|
"go-mod.ewintr.nl/gte/pkg/msend"
|
|
|
|
"go-mod.ewintr.nl/gte/pkg/mstore"
|
2022-10-18 16:58:12 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
type Tasks struct {
|
|
|
|
local storage.LocalRepository
|
|
|
|
remote *storage.RemoteRepository
|
|
|
|
disp *storage.Dispatcher
|
|
|
|
}
|
|
|
|
|
2022-10-23 12:45:21 +02:00
|
|
|
func NewTasks(conf *Configuration, tasks []*task.Task) *Tasks {
|
|
|
|
local := storage.NewMemory(tasks...)
|
2022-10-18 16:58:12 +02:00
|
|
|
remote := storage.NewRemoteRepository(mstore.NewIMAP(conf.IMAP()))
|
|
|
|
disp := storage.NewDispatcher(msend.NewSSLSMTP(conf.SMTP()))
|
|
|
|
|
|
|
|
return &Tasks{
|
|
|
|
local: local,
|
|
|
|
remote: remote,
|
|
|
|
disp: disp,
|
2022-10-23 12:45:21 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (t *Tasks) All() ([]*task.Task, error) {
|
|
|
|
lts, err := t.local.FindAll()
|
|
|
|
if err != nil {
|
|
|
|
return []*task.Task{}, err
|
|
|
|
}
|
|
|
|
for _, lt := range lts {
|
|
|
|
lt.ApplyUpdate()
|
|
|
|
}
|
|
|
|
ts := []*task.Task{}
|
|
|
|
for _, lt := range lts {
|
|
|
|
ts = append(ts, <.Task)
|
|
|
|
}
|
|
|
|
|
|
|
|
return ts, nil
|
2022-10-18 16:58:12 +02:00
|
|
|
}
|
|
|
|
|
2022-10-20 10:40:46 +02:00
|
|
|
func (t *Tasks) Today() (map[string]string, error) {
|
2022-10-18 16:58:12 +02:00
|
|
|
reqs := process.ListReqs{
|
|
|
|
Due: task.Today(),
|
|
|
|
IncludeBefore: true,
|
|
|
|
ApplyUpdates: true,
|
|
|
|
}
|
|
|
|
res, err := process.NewList(t.local, reqs).Process()
|
|
|
|
if err != nil {
|
2022-10-20 10:40:46 +02:00
|
|
|
return map[string]string{}, err
|
2022-10-18 16:58:12 +02:00
|
|
|
}
|
|
|
|
sort.Sort(task.ByDefault(res.Tasks))
|
|
|
|
|
2022-10-20 10:40:46 +02:00
|
|
|
tasks := map[string]string{}
|
2022-10-18 16:58:12 +02:00
|
|
|
for _, t := range res.Tasks {
|
2022-10-20 10:40:46 +02:00
|
|
|
tasks[t.Id] = t.Action
|
2022-10-18 16:58:12 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return tasks, nil
|
|
|
|
}
|
|
|
|
|
2022-10-23 16:46:37 +02:00
|
|
|
func (t *Tasks) Sync() (int, int, error) {
|
2022-10-20 10:40:46 +02:00
|
|
|
countDisp, err := process.NewSend(t.local, t.disp).Process()
|
|
|
|
if err != nil {
|
|
|
|
return 0, 0, err
|
|
|
|
}
|
|
|
|
|
2022-10-25 15:59:09 +02:00
|
|
|
latestFetch, latestDisp, err := t.local.LatestSyncs()
|
2022-10-20 10:40:46 +02:00
|
|
|
if err != nil {
|
|
|
|
return 0, 0, err
|
|
|
|
}
|
2022-10-23 16:46:37 +02:00
|
|
|
// use unix timestamp for time comparison, because time.Before and
|
2022-10-25 15:59:09 +02:00
|
|
|
// time.After depend on a monotonic clock and on my phone the
|
|
|
|
// monotonic clock stops ticking when it goes to suspended sleep
|
|
|
|
if latestFetch.Add(15*time.Minute).Unix() > time.Now().Unix() || latestDisp.Add(2*time.Minute).Unix() > time.Now().Unix() {
|
2022-10-20 10:40:46 +02:00
|
|
|
return countDisp, 0, nil
|
|
|
|
}
|
2022-10-23 16:46:37 +02:00
|
|
|
|
2022-10-25 15:59:09 +02:00
|
|
|
res, err := process.NewFetch(t.remote, t.local, task.FOLDER_PLANNED).Process()
|
2022-10-18 16:58:12 +02:00
|
|
|
if err != nil {
|
2022-10-20 10:40:46 +02:00
|
|
|
return countDisp, 0, err
|
2022-10-18 16:58:12 +02:00
|
|
|
}
|
2022-10-20 10:40:46 +02:00
|
|
|
return countDisp, res.Count, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (t *Tasks) MarkDone(id string) error {
|
|
|
|
localTask, err := t.local.FindById(id)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
update := &task.LocalUpdate{
|
|
|
|
ForVersion: localTask.Version,
|
|
|
|
Fields: []string{task.FIELD_DONE},
|
|
|
|
Done: true,
|
|
|
|
}
|
|
|
|
|
|
|
|
return process.NewUpdate(t.local, id, update).Process()
|
2022-10-18 16:58:12 +02:00
|
|
|
}
|
2022-10-31 09:18:36 +01:00
|
|
|
|
|
|
|
func (t *Tasks) Add(fields map[string]string) error {
|
|
|
|
update := &task.LocalUpdate{
|
|
|
|
Fields: []string{},
|
|
|
|
}
|
|
|
|
if len(fields["action"]) != 0 {
|
|
|
|
update.Action = fields["action"]
|
|
|
|
update.Fields = append(update.Fields, task.FIELD_ACTION)
|
|
|
|
}
|
|
|
|
if len(fields["project"]) != 0 {
|
|
|
|
update.Project = fields["project"]
|
|
|
|
update.Fields = append(update.Fields, task.FIELD_PROJECT)
|
|
|
|
}
|
|
|
|
due := task.NewDateFromString(fields["due"])
|
|
|
|
if !due.IsZero() {
|
|
|
|
update.Due = due
|
|
|
|
update.Fields = append(update.Fields, task.FIELD_DUE)
|
|
|
|
}
|
|
|
|
recur := task.NewRecurrer(fields["recur"])
|
|
|
|
if recur != nil {
|
|
|
|
update.Recur = recur
|
|
|
|
update.Fields = append(update.Fields, task.FIELD_RECUR)
|
|
|
|
}
|
|
|
|
if len(update.Fields) == 0 {
|
|
|
|
return fmt.Errorf("no fields in new task")
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := process.NewNew(t.local, update).Process(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
2022-10-31 15:59:58 +01:00
|
|
|
|
|
|
|
func (t *Tasks) Update(id, newDue string) error {
|
|
|
|
due := task.NewDateFromString(newDue)
|
|
|
|
localTask, err := t.local.FindById(id)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
update := &task.LocalUpdate{
|
|
|
|
ForVersion: localTask.Version,
|
|
|
|
Due: due,
|
|
|
|
Fields: []string{task.FIELD_DUE},
|
|
|
|
}
|
|
|
|
|
|
|
|
return process.NewUpdate(t.local, localTask.Id, update).Process()
|
|
|
|
}
|