package storage import ( "errors" "sort" "time" "go-mod.ewintr.nl/planner/item" ) var ( ErrNotFound = errors.New("not found") ) type Txer interface { Begin() (*Tx, error) } type LocalID interface { FindOne(lid int) (string, error) FindAll() (map[string]int, error) FindOrNext(id string) (int, error) Next() (int, error) Store(id string, localID int) error Delete(id string) error } type Sync interface { FindAll() ([]item.Item, error) Store(i item.Item) error DeleteAll() error SetLastUpdate(ts time.Time) error LastUpdate() (time.Time, error) } type TaskListParams struct { HasRecurrer bool HasDate bool From item.Date To item.Date Project string } type Task interface { Store(task item.Task) error FindOne(id string) (item.Task, error) FindMany(params TaskListParams) ([]item.Task, error) Delete(id string) error Projects() (map[string]int, error) } func Match(tsk item.Task, params TaskListParams) bool { if params.HasRecurrer && tsk.Recurrer == nil { return false } if params.HasDate && tsk.Date.IsZero() { return false } if !params.From.IsZero() && params.From.After(tsk.Date) { return false } if !params.To.IsZero() && tsk.Date.After(params.To) { return false } if params.Project != "" && params.Project != tsk.Project { return false } return true } func NextLocalID(used []int) int { if len(used) == 0 { return 1 } sort.Ints(used) usedMax := 1 for _, u := range used { if u > usedMax { usedMax = u } } var limit int for limit = 1; limit <= len(used) || limit < usedMax; limit *= 10 { } newId := used[len(used)-1] + 1 if newId < limit { return newId } usedMap := map[int]bool{} for _, u := range used { usedMap[u] = true } for i := 1; i < limit; i++ { if _, ok := usedMap[i]; !ok { return i } } return limit }