2018-01-16 10:50:08 +01:00
|
|
|
package main
|
|
|
|
|
2018-01-16 12:06:20 +01:00
|
|
|
import (
|
|
|
|
"database/sql"
|
|
|
|
_ "github.com/mattn/go-sqlite3"
|
|
|
|
"os"
|
|
|
|
"os/user"
|
2018-01-20 11:01:53 +01:00
|
|
|
"strings"
|
2018-01-16 13:02:35 +01:00
|
|
|
"time"
|
2018-01-16 12:06:20 +01:00
|
|
|
)
|
|
|
|
|
2018-01-16 13:02:35 +01:00
|
|
|
// 2018-01-16 19:05:21.752851759+08:00
|
|
|
|
const datetimeFmt = "2006-01-02 15:04:05.999999999-07:00"
|
|
|
|
|
2018-01-16 12:06:20 +01:00
|
|
|
func defaultDBPath() string {
|
|
|
|
u, err := user.Current()
|
|
|
|
maybe(err)
|
|
|
|
return u.HomeDir + "/.pomo"
|
|
|
|
}
|
|
|
|
|
|
|
|
type Store struct {
|
|
|
|
db *sql.DB
|
|
|
|
}
|
2018-01-16 10:50:08 +01:00
|
|
|
|
|
|
|
func NewStore(path string) (*Store, error) {
|
2018-01-16 12:06:20 +01:00
|
|
|
os.Mkdir(path, 0755)
|
|
|
|
db, err := sql.Open("sqlite3", path+"/pomo.db")
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return &Store{db: db}, nil
|
2018-01-16 10:50:08 +01:00
|
|
|
}
|
|
|
|
|
2018-01-16 13:02:35 +01:00
|
|
|
func (s Store) CreateTask(task Task) (int, error) {
|
2018-01-16 12:06:20 +01:00
|
|
|
var taskID int
|
|
|
|
tx, err := s.db.Begin()
|
|
|
|
if err != nil {
|
|
|
|
return -1, err
|
|
|
|
}
|
2018-01-20 11:01:53 +01:00
|
|
|
_, err = tx.Exec("INSERT INTO task (message,tags) VALUES ($1,$2)", task.Message, strings.Join(task.Tags, ","))
|
2018-01-16 12:06:20 +01:00
|
|
|
if err != nil {
|
|
|
|
tx.Rollback()
|
|
|
|
return -1, err
|
|
|
|
}
|
|
|
|
err = tx.QueryRow("SELECT last_insert_rowid() FROM task").Scan(&taskID)
|
|
|
|
if err != nil {
|
|
|
|
tx.Rollback()
|
|
|
|
return -1, err
|
|
|
|
}
|
|
|
|
return taskID, tx.Commit()
|
|
|
|
}
|
2018-01-16 10:50:08 +01:00
|
|
|
|
2018-01-20 13:17:58 +01:00
|
|
|
func (s Store) CreatePomodoro(taskID int, pomodoro Pomodoro) error {
|
2018-01-16 12:06:20 +01:00
|
|
|
_, err := s.db.Exec(
|
2018-01-20 13:17:58 +01:00
|
|
|
`INSERT INTO pomodoro (task_id, start, end) VALUES ($1, $2, $3)`,
|
2018-01-16 12:06:20 +01:00
|
|
|
taskID,
|
2018-01-20 13:17:58 +01:00
|
|
|
pomodoro.Start,
|
|
|
|
pomodoro.End,
|
2018-01-16 12:06:20 +01:00
|
|
|
)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2018-01-16 13:02:35 +01:00
|
|
|
func (s Store) ReadTasks() ([]*Task, error) {
|
2018-01-20 11:01:53 +01:00
|
|
|
rows, err := s.db.Query(`SELECT rowid,message,tags FROM task`)
|
2018-01-16 13:02:35 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
tasks := []*Task{}
|
|
|
|
for rows.Next() {
|
2018-01-20 11:01:53 +01:00
|
|
|
var tags string
|
2018-01-20 13:17:58 +01:00
|
|
|
task := &Task{Pomodoros: []*Pomodoro{}}
|
2018-01-20 11:01:53 +01:00
|
|
|
err = rows.Scan(&task.ID, &task.Message, &tags)
|
2018-01-16 13:02:35 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2018-01-20 13:19:14 +01:00
|
|
|
if tags != "" {
|
|
|
|
task.Tags = strings.Split(tags, ",")
|
|
|
|
}
|
2018-01-20 13:17:58 +01:00
|
|
|
pomodoros, err := s.ReadPomodoros(task.ID)
|
2018-01-16 13:02:35 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2018-01-20 13:17:58 +01:00
|
|
|
for _, pomodoro := range pomodoros {
|
|
|
|
task.Pomodoros = append(task.Pomodoros, pomodoro)
|
2018-01-16 13:02:35 +01:00
|
|
|
}
|
|
|
|
tasks = append(tasks, task)
|
|
|
|
}
|
|
|
|
return tasks, nil
|
|
|
|
}
|
|
|
|
|
2018-01-20 13:17:58 +01:00
|
|
|
func (s Store) ReadPomodoros(taskID int) ([]*Pomodoro, error) {
|
|
|
|
rows, err := s.db.Query(`SELECT start,end FROM pomodoro WHERE task_id = $1`, &taskID)
|
2018-01-16 13:02:35 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2018-01-20 13:17:58 +01:00
|
|
|
pomodoros := []*Pomodoro{}
|
2018-01-16 13:02:35 +01:00
|
|
|
for rows.Next() {
|
|
|
|
var (
|
|
|
|
startStr string
|
|
|
|
endStr string
|
|
|
|
)
|
2018-01-20 13:17:58 +01:00
|
|
|
pomodoro := &Pomodoro{}
|
2018-01-16 13:02:35 +01:00
|
|
|
err = rows.Scan(&startStr, &endStr)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
start, _ := time.Parse(datetimeFmt, startStr)
|
|
|
|
end, _ := time.Parse(datetimeFmt, endStr)
|
2018-01-20 13:17:58 +01:00
|
|
|
pomodoro.Start = start
|
|
|
|
pomodoro.End = end
|
|
|
|
pomodoros = append(pomodoros, pomodoro)
|
2018-01-16 13:02:35 +01:00
|
|
|
}
|
2018-01-20 13:17:58 +01:00
|
|
|
return pomodoros, nil
|
2018-01-16 13:02:35 +01:00
|
|
|
}
|
|
|
|
|
2018-01-17 08:53:45 +01:00
|
|
|
func (s Store) DeleteTask(taskID int) error {
|
|
|
|
tx, err := s.db.Begin()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
_, err = tx.Exec("DELETE FROM task WHERE rowid = $1", &taskID)
|
|
|
|
if err != nil {
|
|
|
|
tx.Rollback()
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
_, err = tx.Exec("DELETE FROM record WHERE task_id = $1", &taskID)
|
|
|
|
if err != nil {
|
|
|
|
tx.Rollback()
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return tx.Commit()
|
|
|
|
}
|
|
|
|
|
2018-01-16 12:06:20 +01:00
|
|
|
func (s Store) Close() error { return s.db.Close() }
|
|
|
|
|
|
|
|
func initDB(db *Store) error {
|
|
|
|
stmt := `
|
|
|
|
CREATE TABLE task (
|
2018-01-20 11:01:53 +01:00
|
|
|
message TEXT,
|
|
|
|
tags TEXT
|
2018-01-16 12:06:20 +01:00
|
|
|
);
|
2018-01-20 13:17:58 +01:00
|
|
|
CREATE TABLE pomodoro (
|
2018-01-16 12:06:20 +01:00
|
|
|
task_id INTEGER,
|
|
|
|
start DATETTIME,
|
|
|
|
end DATETTIME
|
|
|
|
);
|
|
|
|
`
|
|
|
|
_, err := db.db.Exec(stmt)
|
|
|
|
return err
|
|
|
|
}
|