gime/cli/cli.go
2022-01-21 16:49:05 -06:00

182 lines
4.1 KiB
Go

package cli
import (
"fmt"
"strings"
"time"
"git.bullercodeworks.com/brian/gime/util"
timertxt "git.bullercodeworks.com/brian/go-timertxt"
"github.com/spf13/viper"
)
type Program struct {
debug bool
timerPath string
TimerList *timertxt.TimerList
donePath string
DoneList *timertxt.TimerList
}
func (p *Program) Initialize() error {
path := viper.GetString("directory")
p.timerPath = fmt.Sprintf("%s%s", path, viper.GetString("timerfile"))
p.donePath = fmt.Sprintf("%s%s", path, viper.GetString("donefile"))
return nil
}
func (p *Program) GetTimerFilePath() string {
return p.timerPath
}
func (p *Program) LoadTimerList() error {
var err error
var tl timertxt.TimerList
tl, err = timertxt.LoadFromFilename(p.timerPath)
if err != nil {
return err
}
tl.Sort(timertxt.SORT_UNFINISHED_START)
p.TimerList = &tl
return nil
}
func (p *Program) WriteTimerList() error {
return p.TimerList.WriteToFilename(p.timerPath)
}
func (p *Program) GetDoneFilePath() string {
return p.donePath
}
func (p *Program) LoadDoneList() error {
var err error
var tl timertxt.TimerList
tl, err = timertxt.LoadFromFilename(p.donePath)
if err != nil {
return err
}
p.DoneList = &tl
return nil
}
func (p *Program) WriteDoneList() error {
return p.DoneList.WriteToFilename(p.donePath)
}
func (p *Program) ArchiveTimer(timer *timertxt.Timer) error {
if err := p.TimerList.ArchiveTimerToFile(*timer, p.donePath); err != nil {
return err
}
return p.WriteTimerList()
}
func (p *Program) GetFilteredTimerList(args []string) *timertxt.TimerList {
var includeArchive bool
var err error
start := time.Time{}
end := time.Now()
var contextFilters []string
var projectFilters []string
var allFilters []func(timertxt.Timer) bool
if len(args) > 0 {
contextFilters, args = util.GetContextsFromSlice(args)
projectFilters, args = util.GetProjectsFromSlice(args)
}
if len(args) > 0 {
if args[0] == "--a" {
includeArchive = true
args = args[1:]
}
}
if len(args) > 0 {
if start, err = util.ParseFuzzyTime(args[0]); err != nil {
y, m, d := time.Now().Date()
start = time.Date(y, m, d, 0, 0, 0, 0, time.Now().Location())
} else {
args = args[1:]
}
if len(args) > 0 {
if end, err = util.ParseFuzzyTime(args[0]); err != nil {
y, m, d := time.Now().Date()
end = time.Date(y, m, d, 23, 59, 59, 0, time.Now().Location())
} else {
args = args[1:]
}
}
}
list := p.TimerList.GetTimersInRange(start, end)
if includeArchive {
if err = p.LoadDoneList(); err != nil {
fmt.Println("Error loading done.txt entries")
fmt.Println(err.Error())
return nil
}
*list = append(*list, (*p.DoneList.GetTimersInRange(start, end))...)
}
if len(contextFilters) > 0 {
allFilters = append(allFilters, func(t timertxt.Timer) bool {
for _, v := range contextFilters {
v = strings.TrimPrefix(v, "@")
if !t.HasContext(v) {
return false
}
}
return true
})
}
if len(projectFilters) > 0 {
allFilters = append(allFilters, func(t timertxt.Timer) bool {
for _, v := range projectFilters {
v = strings.TrimPrefix(v, "+")
if !t.HasProject(v) {
return false
}
}
return true
})
}
doFilters := func(t *timertxt.Timer) bool {
for _, v := range allFilters {
if !v(*t) {
return false
}
}
// If we made it all the way down here, it matches
return true
}
return list.Filter(doFilters)
}
func (p *Program) GetMostRecentTimer() (*timertxt.Timer, error) {
work, wErr := p.TimerList.GetMostRecentTimer()
if wErr == nil && work.FinishDate.IsZero() {
return work, nil
}
if err := p.LoadDoneList(); err != nil {
return nil, err
}
done, dErr := p.DoneList.GetMostRecentTimer()
if dErr != nil {
return nil, dErr
}
if !done.FinishDate.IsZero() && work == nil || done.FinishDate.After(work.FinishDate) {
return done, nil
}
return work, nil
}
func (p *Program) GetTimerString(timer *timertxt.Timer) string {
return fmt.Sprintf("%d. %s", timer.Id, strings.TrimPrefix(timer.String(), "x "))
}
func (p *Program) GetDoneTimerString(timer *timertxt.Timer) string {
return fmt.Sprintf("--. %s", timer.String())
}