kevinschoon-pomo/pkg/cmd/cmd.go

257 lines
6.4 KiB
Go

package cmd
import (
"database/sql"
"encoding/json"
"fmt"
"os"
"os/user"
"path"
"sort"
"time"
cli "github.com/jawher/mow.cli"
pomo "github.com/kevinschoon/pomo/pkg/internal"
)
func maybe(err error) {
if err != nil {
fmt.Printf("Error:\n%s\n", err)
os.Exit(1)
}
}
func defaultConfigPath() string {
u, err := user.Current()
maybe(err)
return path.Join(u.HomeDir, "/.pomo/config.json")
}
func start(config *pomo.Config) func(*cli.Cmd) {
return func(cmd *cli.Cmd) {
cmd.Spec = "[OPTIONS] MESSAGE"
var (
duration = cmd.StringOpt("d duration", "25m", "duration of each stent")
pomodoros = cmd.IntOpt("p pomodoros", 4, "number of pomodoros")
message = cmd.StringArg("MESSAGE", "", "descriptive name of the given task")
tags = cmd.StringsOpt("t tag", []string{}, "tags associated with this task")
)
cmd.Action = func() {
parsed, err := time.ParseDuration(*duration)
maybe(err)
db, err := pomo.NewStore(config.DBPath)
maybe(err)
defer db.Close()
task := &pomo.Task{
Message: *message,
Tags: *tags,
NPomodoros: *pomodoros,
Duration: parsed,
}
maybe(db.With(func(tx *sql.Tx) error {
id, err := db.CreateTask(tx, *task)
if err != nil {
return err
}
task.ID = id
return nil
}))
runner, err := pomo.NewTaskRunner(task, config)
maybe(err)
server, err := pomo.NewServer(runner, config)
maybe(err)
server.Start()
defer server.Stop()
runner.Start()
pomo.StartUI(runner)
}
}
}
func create(config *pomo.Config) func(*cli.Cmd) {
return func(cmd *cli.Cmd) {
cmd.Spec = "[OPTIONS] MESSAGE"
var (
duration = cmd.StringOpt("d duration", "25m", "duration of each stent")
pomodoros = cmd.IntOpt("p pomodoros", 4, "number of pomodoros")
message = cmd.StringArg("MESSAGE", "", "descriptive name of the given task")
tags = cmd.StringsOpt("t tag", []string{}, "tags associated with this task")
)
cmd.Action = func() {
parsed, err := time.ParseDuration(*duration)
maybe(err)
db, err := pomo.NewStore(config.DBPath)
maybe(err)
defer db.Close()
task := &pomo.Task{
Message: *message,
Tags: *tags,
NPomodoros: *pomodoros,
Duration: parsed,
}
maybe(db.With(func(tx *sql.Tx) error {
taskId, err := db.CreateTask(tx, *task)
if err != nil {
return err
}
fmt.Println(taskId)
return nil
}))
}
}
}
func begin(config *pomo.Config) func(*cli.Cmd) {
return func(cmd *cli.Cmd) {
cmd.Spec = "[OPTIONS] TASK_ID"
var (
taskId = cmd.IntArg("TASK_ID", -1, "ID of Pomodoro to begin")
)
cmd.Action = func() {
db, err := pomo.NewStore(config.DBPath)
maybe(err)
defer db.Close()
var task *pomo.Task
maybe(db.With(func(tx *sql.Tx) error {
read, err := db.ReadTask(tx, *taskId)
if err != nil {
return err
}
task = read
err = db.DeletePomodoros(tx, *taskId)
if err != nil {
return err
}
task.Pomodoros = []*pomo.Pomodoro{}
return nil
}))
runner, err := pomo.NewTaskRunner(task, config)
maybe(err)
server, err := pomo.NewServer(runner, config)
maybe(err)
server.Start()
defer server.Stop()
runner.Start()
pomo.StartUI(runner)
}
}
}
func initialize(config *pomo.Config) func(*cli.Cmd) {
return func(cmd *cli.Cmd) {
cmd.Spec = "[OPTIONS]"
cmd.Action = func() {
db, err := pomo.NewStore(config.DBPath)
maybe(err)
defer db.Close()
maybe(pomo.InitDB(db))
}
}
}
func list(config *pomo.Config) func(*cli.Cmd) {
return func(cmd *cli.Cmd) {
cmd.Spec = "[OPTIONS]"
var (
asJSON = cmd.BoolOpt("json", false, "output task history as JSON")
assend = cmd.BoolOpt("assend", false, "sort tasks assending in age")
all = cmd.BoolOpt("a all", true, "output all tasks")
limit = cmd.IntOpt("n limit", 0, "limit the number of results by n")
duration = cmd.StringOpt("d duration", "24h", "show tasks within this duration")
)
cmd.Action = func() {
duration, err := time.ParseDuration(*duration)
maybe(err)
db, err := pomo.NewStore(config.DBPath)
maybe(err)
defer db.Close()
maybe(db.With(func(tx *sql.Tx) error {
tasks, err := db.ReadTasks(tx)
maybe(err)
if *assend {
sort.Sort(sort.Reverse(pomo.ByID(tasks)))
}
if !*all {
tasks = pomo.After(time.Now().Add(-duration), tasks)
}
if *limit > 0 && (len(tasks) > *limit) {
tasks = tasks[0:*limit]
}
if *asJSON {
maybe(json.NewEncoder(os.Stdout).Encode(tasks))
return nil
}
maybe(err)
pomo.SummerizeTasks(config, tasks)
return nil
}))
}
}
}
func _delete(config *pomo.Config) func(*cli.Cmd) {
return func(cmd *cli.Cmd) {
cmd.Spec = "[OPTIONS] TASK_ID"
var taskID = cmd.IntArg("TASK_ID", -1, "task to delete")
cmd.Action = func() {
db, err := pomo.NewStore(config.DBPath)
maybe(err)
defer db.Close()
maybe(db.With(func(tx *sql.Tx) error {
return db.DeleteTask(tx, *taskID)
}))
}
}
}
func _status(config *pomo.Config) func(*cli.Cmd) {
return func(cmd *cli.Cmd) {
cmd.Spec = "[OPTIONS]"
cmd.Action = func() {
client, err := pomo.NewClient(config.SocketPath)
if err != nil {
pomo.OutputStatus(pomo.Status{})
return
}
defer client.Close()
status, err := client.Status()
maybe(err)
pomo.OutputStatus(*status)
}
}
}
func _config(config *pomo.Config) func(*cli.Cmd) {
return func(cmd *cli.Cmd) {
cmd.Spec = "[OPTIONS]"
cmd.Action = func() {
maybe(json.NewEncoder(os.Stdout).Encode(config))
}
}
}
func Run() {
app := cli.App("pomo", "Pomodoro CLI")
app.LongDesc = "Pomo helps you track what you did, how long it took you to do it, and how much effort you expect it to take."
app.Spec = "[OPTIONS]"
var (
config = &pomo.Config{}
path = app.StringOpt("p path", defaultConfigPath(), "path to the pomo config directory")
)
app.Before = func() {
maybe(pomo.LoadConfig(*path, config))
}
app.Version("v version", pomo.Version)
app.Command("start s", "start a new task", start(config))
app.Command("init", "initialize the sqlite database", initialize(config))
app.Command("config cf", "display the current configuration", _config(config))
app.Command("create c", "create a new task without starting", create(config))
app.Command("begin b", "begin requested pomodoro", begin(config))
app.Command("list l", "list historical tasks", list(config))
app.Command("delete d", "delete a stored task", _delete(config))
app.Command("status st", "output the current status", _status(config))
app.Run(os.Args)
}