gte/internal/storage/remote.go

159 lines
3.3 KiB
Go

package storage
import (
"errors"
"fmt"
"strconv"
"ewintr.nl/gte/internal/task"
"ewintr.nl/gte/pkg/mstore"
)
var (
ErrMStoreError = errors.New("mstore gave error response")
ErrInvalidTask = errors.New("invalid task")
ErrInvalidMessage = errors.New("task contains invalid message")
)
type RemoteRepository struct {
mstore mstore.MStorer
}
func NewRemoteRepository(ms mstore.MStorer) *RemoteRepository {
return &RemoteRepository{
mstore: ms,
}
}
func (rr *RemoteRepository) FindAll(folder string) ([]*task.Task, error) {
msgs, err := rr.mstore.Messages(folder)
if err != nil {
return []*task.Task{}, fmt.Errorf("%w: %v", ErrMStoreError, err)
}
tasks := []*task.Task{}
for _, msg := range msgs {
if msg.Valid() {
tasks = append(tasks, task.NewFromMessage(msg))
}
}
return tasks, nil
}
func (rr *RemoteRepository) Update(t *task.Task) error {
if t == nil {
return ErrInvalidTask
}
// add new
if err := rr.Add(t); err != nil {
return err
}
// remove old
if err := rr.mstore.Remove(t.Message); err != nil {
return fmt.Errorf("%w: %s", ErrMStoreError, err)
}
return nil
}
func (rr *RemoteRepository) Add(t *task.Task) error {
if t == nil {
return ErrInvalidTask
}
msg := t.NextMessage()
if err := rr.mstore.Add(msg.Folder, msg.Subject, msg.Body); err != nil {
return fmt.Errorf("%w: %v", ErrMStoreError, err)
}
return nil
}
// Cleanup removes older versions of tasks
func (rr *RemoteRepository) CleanUp() error {
// loop through folders, get all task version info
type msgInfo struct {
Version int
Message *mstore.Message
}
msgsSet := make(map[string][]msgInfo)
for _, folder := range task.KnownFolders {
msgs, err := rr.mstore.Messages(folder)
if err != nil {
return fmt.Errorf("%w: %v", ErrMStoreError, err)
}
for _, msg := range msgs {
id, _ := task.FieldFromBody(task.FIELD_ID, msg.Body)
versionStr, _ := task.FieldFromBody(task.FIELD_VERSION, msg.Body)
version, _ := strconv.Atoi(versionStr)
if _, ok := msgsSet[id]; !ok {
msgsSet[id] = []msgInfo{}
}
msgsSet[id] = append(msgsSet[id], msgInfo{
Version: version,
Message: msg,
})
}
}
// determine which ones need to be gone
var tobeRemoved []*mstore.Message
for _, mInfos := range msgsSet {
maxVersion := 0
for _, mInfo := range mInfos {
if mInfo.Version > maxVersion {
maxVersion = mInfo.Version
}
}
for _, mInfo := range mInfos {
if mInfo.Version < maxVersion {
tobeRemoved = append(tobeRemoved, mInfo.Message)
}
}
}
// remove them
for _, msg := range tobeRemoved {
if err := rr.mstore.Remove(msg); err != nil {
return err
}
}
return nil
}
func (rr *RemoteRepository) Remove(tasks []*task.Task) error {
tMap := map[string]*task.Task{}
for _, t := range tasks {
tMap[t.Id] = t
}
var toBeRemoved []*mstore.Message
for _, folder := range task.KnownFolders {
msgs, err := rr.mstore.Messages(folder)
if err != nil {
return fmt.Errorf("%w: %v", ErrMStoreError, err)
}
for _, msg := range msgs {
id, _ := task.FieldFromBody(task.FIELD_ID, msg.Body)
if _, ok := tMap[id]; ok {
toBeRemoved = append(toBeRemoved, msg)
}
}
}
for _, msg := range toBeRemoved {
if err := rr.mstore.Remove(msg); err != nil {
return fmt.Errorf("%w: %v", ErrMStoreError, err)
}
}
return nil
}