24 Commits

Author SHA1 Message Date
Bel LaPointe
ccdd2615c4 instead of trashing editing file in cli, write err msg and re-edit 2022-01-04 12:37:35 -05:00
Bel LaPointe
0241f2d76c err msg 2022-01-04 12:33:39 -05:00
Bel LaPointe
fa812a16ee fix checking hastags and nothas tags 2022-01-04 08:56:19 -05:00
Bel LaPointe
cf3b233a11 add anti tag search 2022-01-04 07:01:35 -05:00
Bel LaPointe
64868bdc0b create script to scrape list of lists and scrape aech into a yaml 2022-01-03 16:36:59 -05:00
Bel LaPointe
1b1724bea7 update todo with no more ts in todo section 2022-01-02 20:45:17 -05:00
Bel LaPointe
f4968d1d1b update todo 2022-01-02 20:44:23 -05:00
Bel LaPointe
6203ced79b root does not marshal TSs on todos, always on schedules, dones. Test. 2022-01-02 20:44:04 -05:00
Bel LaPointe
4d171d10d0 only write TS on a todo if writeTS is set 2022-01-02 20:37:40 -05:00
Bel LaPointe
8ad2ca9345 new todo 2022-01-02 20:32:16 -05:00
Bel LaPointe
bc0bb05fb6 update todo 2022-01-02 20:31:09 -05:00
Bel LaPointe
22dc04341c update todo.yaml 2022-01-02 20:17:32 -05:00
Bel LaPointe
bf10c00708 add cli -g to merge in a file 2022-01-02 20:07:52 -05:00
Bel LaPointe
cd5ce0f0df add root MergeIn 2022-01-01 17:43:20 -05:00
Bel LaPointe
eb41f065a5 todo 2022-01-01 17:33:59 -05:00
Bel LaPointe
96bfb96ee3 consts for what to print, default to just the todos 2022-01-01 17:30:40 -05:00
Bel LaPointe
770f2719d2 ezdate for yyyy-mm-dd for schedule 2022-01-01 17:21:38 -05:00
Bel LaPointe
3554bf8ba4 rm test code 2022-01-01 17:14:03 -05:00
Bel LaPointe
b4aa4ad310 ts shouldnt yield zero ever, yield now if so 2022-01-01 17:13:50 -05:00
Bel LaPointe
0dddf26265 fix getting ts for completed tasks 2022-01-01 17:04:21 -05:00
Bel LaPointe
5afdeed3b5 todo 2021-12-31 23:27:21 -05:00
Bel LaPointe
af0f094a65 support tag, simple case insensitve search when recursing 2021-12-31 23:20:27 -05:00
Bel LaPointe
031db8788b tag saerch on todo 2021-12-31 23:14:31 -05:00
Bel LaPointe
b8efdbfa52 when writing output file, dont recurse 2021-12-31 23:03:08 -05:00
10 changed files with 456 additions and 49 deletions

View File

@@ -11,11 +11,19 @@ import (
"os" "os"
"os/exec" "os/exec"
"path" "path"
"strings"
"syscall" "syscall"
"gopkg.in/yaml.v2" "gopkg.in/yaml.v2"
) )
const (
DUMP_ALL = "all"
DUMP_TODO = "todo"
DUMP_SCHEDULED = "scheduled"
DUMP_DONE = "done"
)
func main() { func main() {
if err := _main(); err != nil { if err := _main(); err != nil {
panic(err) panic(err)
@@ -28,15 +36,53 @@ func _main() error {
defaultFilepath = "-" defaultFilepath = "-"
} }
filepath := flag.String("f", defaultFilepath, "($PTTODO_FILE) path to yaml file") filepath := flag.String("f", defaultFilepath, "($PTTODO_FILE) path to yaml file")
filepathToMergeIn := flag.String("g", "", "path to yaml file to merge into -f")
root := flag.String("root", DUMP_TODO, "path to pretty print ("+fmt.Sprint([]string{DUMP_ALL, DUMP_TODO, DUMP_SCHEDULED, DUMP_DONE})+")")
tags := flag.String("tags", "", "csv of all tags to find, -tag to invert")
search := flag.String("search", "", "fts case insensitive")
e := flag.Bool("e", false, "edit file") e := flag.Bool("e", false, "edit file")
dry := flag.Bool("dry", false, "dry run") dry := flag.Bool("dry", false, "dry run")
flag.Parse() flag.Parse()
if *filepathToMergeIn != "" {
if err := merge(*dry, *filepath, *filepathToMergeIn); err != nil {
return err
}
}
if *e { if *e {
if err := edit(*dry, *filepath); err != nil { if err := edit(*dry, *filepath); err != nil {
return err return err
} }
} }
return dump(*dry, os.Stdout, *filepath) var tagslist []string
if *tags != "" {
tagslist = strings.Split(*tags, ",")
}
return dump(*dry, os.Stdout, *filepath, tagslist, *search, *root)
}
func verifyRoot(root pttodo.Root) error {
f, err := ioutil.TempFile(os.TempDir(), "tmp")
if err != nil {
return err
}
f.Close()
tempFile := f.Name()
b, err := yaml.Marshal(root)
if err != nil {
return err
}
if err := ioutil.WriteFile(tempFile, b, os.ModePerm); err != nil {
return err
}
defer os.Remove(tempFile)
return verifyFile(tempFile)
}
func verifyFile(path string) error {
if err := dump(true, io.Discard, path, nil, "", DUMP_ALL); err != nil {
return fmt.Errorf("failed verifying file %s: %w", path, err)
}
return nil
} }
func edit(dry bool, filepath string) error { func edit(dry bool, filepath string) error {
@@ -100,10 +146,21 @@ func edit(dry bool, filepath string) error {
return nil return nil
} }
verify := func() error { verify := func() error {
if err := dump(true, io.Discard, tempFile); err != nil { for {
return fmt.Errorf("failed to verify %s: %v", tempFile, err) err := verifyFile(tempFile)
if err == nil {
break
} }
return nil log.Printf("%v, press <Enter> to resume editing", err)
b := make([]byte, 1)
if _, err := os.Stdin.Read(b); err != nil {
break
}
if err := vi(); err != nil {
return err
}
}
return verifyFile(tempFile)
} }
save := func() error { save := func() error {
if dry { if dry {
@@ -125,16 +182,69 @@ func edit(dry bool, filepath string) error {
return nil return nil
} }
func dump(dry bool, writer io.Writer, filepath string) error { func merge(dry bool, filepath string, mergeTargetFilePath string) error {
var reader io.Reader baseReader, err := filePathReader(filepath)
if filepath == "-" {
reader = os.Stdin
} else {
b, err := ioutil.ReadFile(filepath)
if err != nil { if err != nil {
return err return err
} }
reader = bytes.NewReader(b) baseB, err := ioutil.ReadAll(baseReader)
if err != nil {
return err
}
mergingReader, err := filePathReader(mergeTargetFilePath)
if err != nil {
return err
}
mergingB, err := ioutil.ReadAll(mergingReader)
if err != nil {
return err
}
var base, merging pttodo.Root
if err := yaml.Unmarshal(baseB, &base); err != nil {
return err
}
if err := yaml.Unmarshal(mergingB, &merging); err != nil {
return err
}
base.MergeIn(merging)
if err := verifyRoot(base); err != nil {
return err
}
tmppath, err := marshalRootToTempFile(base)
if err != nil {
return err
}
if dry {
log.Printf("would've moved %s to %s when adding %s", tmppath, filepath, mergeTargetFilePath)
return nil
}
return os.Rename(tmppath, filepath)
}
func marshalRootToTempFile(root pttodo.Root) (string, error) {
f, err := ioutil.TempFile(os.TempDir(), "tmp")
if err != nil {
return "", err
}
f.Close()
os.Remove(f.Name())
b, err := yaml.Marshal(root)
if err != nil {
return "", err
}
filepath := f.Name() + ".yaml"
err = ioutil.WriteFile(filepath, b, os.ModePerm)
return filepath, err
}
func dump(dry bool, writer io.Writer, filepath string, tags []string, search, rootDisplay string) error {
reader, err := filePathReader(filepath)
if err != nil {
return err
} }
b, err := ioutil.ReadAll(reader) b, err := ioutil.ReadAll(reader)
@@ -149,15 +259,43 @@ func dump(dry bool, writer io.Writer, filepath string) error {
root.MoveScheduledToTodo() root.MoveScheduledToTodo()
var v interface{} = root var v interface{} = root
switch flag.Arg(0) { switch rootDisplay {
case "": case DUMP_ALL:
case "todo": case DUMP_TODO:
v = root.Todo v = root.Todo
case "scheduled": case DUMP_SCHEDULED:
v = root.Scheduled v = root.Scheduled
case "done": case DUMP_DONE:
v = root.Done v = root.Done
} }
if todos, ok := v.([]pttodo.Todo); ok {
if len(tags) > 0 {
result := make([]pttodo.Todo, 0, len(todos))
for _, todo := range todos {
skip := false
for _, tag := range tags {
positiveTag := strings.TrimLeft(tag, "-")
hasTag := strings.Contains(todo.Tags, positiveTag)
wantToHaveTag := !strings.HasPrefix(tag, "-")
skip = skip || !(hasTag == wantToHaveTag)
}
if !skip {
result = append(result, todo)
}
}
todos = result
}
if len(search) > 0 {
result := make([]pttodo.Todo, 0, len(todos))
for _, todo := range todos {
if strings.Contains(strings.ToLower(fmt.Sprint(todo)), strings.ToLower(search)) {
result = append(result, todo)
}
}
todos = result
}
v = todos
}
b2, err := yaml.Marshal(v) b2, err := yaml.Marshal(v)
if err != nil { if err != nil {
return err return err
@@ -168,5 +306,23 @@ func dump(dry bool, writer io.Writer, filepath string) error {
return nil return nil
} }
return os.WriteFile(filepath, b2, os.ModePerm) b3, err := yaml.Marshal(root)
if err != nil {
return err
}
return os.WriteFile(filepath, b3, os.ModePerm)
}
func filePathReader(path string) (io.Reader, error) {
var reader io.Reader
if path == "-" {
reader = os.Stdin
} else {
b, err := ioutil.ReadFile(path)
if err != nil {
return nil, err
}
reader = bytes.NewReader(b)
}
return reader, nil
} }

View File

@@ -0,0 +1,21 @@
#! /bin/bash
main() {
cd "$(dirname "$BASH_SOURCE")"
source ./from_todo_server_to_pttodo.sh
type from_todo_server_to_pttodo_main
list_lists | jq .list[] | jq -c . | while read -r line; do
local name="$(echo "$line" | jq -r .name | sed 's/^Today$/todo/g' | tr '[:upper:]' '[:lower:]' | sed 's/ /-/g')"
local id="$(echo "$line" | jq -r .id)"
echo $name, $id
TODO_SERVER_LIST="$id" from_todo_server_to_pttodo_main > ./$name.yaml
done
}
list_lists() {
todo_server_curl "$TODO_SERVER_URL/ajax.php?loadLists=&rnd=0.9900282499544026"
}
if [ "$0" == "$BASH_SOURCE" ]; then
main "$@"
fi

View File

@@ -0,0 +1,55 @@
#! /bin/bash
export TODO_SERVER_URL="${TODO_SERVER_URL:-"https://todo-server.remote.blapointe.com"}"
export TODO_SERVER_HEADERS="${TODO_SERVER_HEADERS:-"Cookie: BOAuthZ=$TODO_SERVER_BOAUTHZ"}"
export TODO_SERVER_LIST="${TODO_SERVER_LIST:-"2548023766"}"
main() {
from_todo_server_to_pttodo_main "$@"
}
from_todo_server_to_pttodo_main() {
set -e
set -o pipefail
local tasks_in_todo="$(fetch_tasks_in_list | format_tasks_in_list)"
local schedule_tasks_in_flight="$(COMPL=1 LOOPING=1 fetch_tasks_in_list | format_tasks_in_list)"
echo "{\"todo\": $tasks_in_todo, \"scheduled\": $schedule_tasks_in_flight, \"done\": []}" | yq -P eval -
}
format_tasks_in_list() {
jq -c .list[] | while read -r line; do
echo "$line" \
| jq '{
todo: .title,
details:.note,
ts: (if .compl == 1 then (.dateCompleted | strptime("%d %b %Y %I:%M %p") | mktime) else .dateEditedInt end),
subtasks: [],
tags: .tags,
schedule: (if (.cron != "") then (.cron) else (.loop) end)
}' \
| jq -c .
done | jq -sc | yq -P eval - | grep -v -E ' (""|\[]|0s)$' | yq -j eval - | jq -c .
}
fetch_tasks_in_list() {
todo_server_curl "$TODO_SERVER_URL/ajax.php?loadTasks=&list=$TODO_SERVER_LIST&compl=${COMPL:-0}&looping=${LOOPING:-0}"
}
todo_server_curl() {
local csv_headers="$TODO_SERVER_HEADERS"
local headers=()
while [ "$csv_headers" != "" ]; do
header="${csv_headers%%,*}"
headers+=("-H" "${header%%:*}: ${header#*:}")
if echo "$csv_headers" | grep -q ,; then
csv_headers="${csv_headers#*,}"
else
csv_headers=""
fi
done
curl -sS "${headers[@]}" "$@"
}
if [ "$0" == "$BASH_SOURCE" ]; then
main "$@"
fi

View File

@@ -18,3 +18,35 @@ func (root *Root) MoveScheduledToTodo() {
} }
} }
} }
func (root *Root) MergeIn(root2 Root) {
for _, listPair := range [][2]*[]Todo{
[2]*[]Todo{&root.Todo, &root2.Todo},
[2]*[]Todo{&root.Scheduled, &root2.Scheduled},
[2]*[]Todo{&root.Done, &root2.Done},
} {
for _, candidate := range *listPair[1] {
found := false
for i := range *listPair[0] {
found = found || ((*listPair[0])[i].Todo == candidate.Todo)
}
if !found {
*listPair[0] = append(*listPair[0], candidate)
}
}
}
}
func (root Root) MarshalYAML() (interface{}, error) {
for i := range root.Todo {
root.Todo[i].writeTS = false
}
for i := range root.Scheduled {
root.Scheduled[i].writeTS = true
}
for i := range root.Done {
root.Done[i].writeTS = true
}
type Alt Root
return (Alt)(root), nil
}

View File

@@ -4,8 +4,11 @@ import (
"encoding/json" "encoding/json"
"fmt" "fmt"
"strconv" "strconv"
"strings"
"testing" "testing"
"time" "time"
yaml "gopkg.in/yaml.v2"
) )
func TestJSONRoot(t *testing.T) { func TestJSONRoot(t *testing.T) {
@@ -96,3 +99,75 @@ func TestRootMoveScheduledToTodo(t *testing.T) {
}) })
} }
} }
func TestMergeRoots(t *testing.T) {
root0yaml := `
todo:
- a
- b
- todo: c
- todo: d
tags: a
- exclusive to 0
`
root1yaml := `
todo:
- a
- b
- todo: c
- todo: d
tags: b
- exclusive to 1
`
rootWantyaml := `
todo:
- a
- b
- todo: c
- todo: d
tags: a
- exclusive to 0
- exclusive to 1
`
var root0, root1, rootWant Root
if err := yaml.Unmarshal([]byte(root0yaml), &root0); err != nil {
t.Fatal(err)
}
if err := yaml.Unmarshal([]byte(root1yaml), &root1); err != nil {
t.Fatal(err)
}
if err := yaml.Unmarshal([]byte(rootWantyaml), &rootWant); err != nil {
t.Fatal(err)
}
root0.MergeIn(root1)
if fmt.Sprintf("%+v", root0) != fmt.Sprintf("%+v", rootWant) {
t.Fatalf("want \n\t%+v, got \n\t%+v", rootWant, root0)
}
}
func TestRootMarshalYAMLWriteTS(t *testing.T) {
root := Root{
Todo: []Todo{Todo{Todo: "todo", TS: 1, writeTS: true}},
Scheduled: []Todo{Todo{Todo: "sched", TS: 2, writeTS: false, Schedule: "2099-01-01"}},
Done: []Todo{Todo{Todo: "done", TS: 3, writeTS: false}},
}
got, err := yaml.Marshal(root)
if err != nil {
t.Fatal(err)
}
if strings.TrimSpace(string(got)) != strings.TrimSpace(`
todo:
- todo
scheduled:
- todo: sched
schedule: "2099-01-01"
ts: Wed Dec 31 19:00:02 EST 1969
done:
- todo: done
ts: Wed Dec 31 19:00:03 EST 1969
`) {
t.Fatal(string(got))
}
}

View File

@@ -37,6 +37,8 @@ func schedulerFactory(s string) scheduler {
} else if scheduleDuePattern.MatchString(s) { } else if scheduleDuePattern.MatchString(s) {
n, _ := strconv.Atoi(s) n, _ := strconv.Atoi(s)
return scheduleDue(n) return scheduleDue(n)
} else if scheduleEZDatePattern.MatchString(s) {
return scheduleEZDate(s)
} }
return scheduleCron(s) return scheduleCron(s)
} }
@@ -88,3 +90,12 @@ var scheduleDuePattern = regexp.MustCompile(`^[0-9]+$`)
func (due scheduleDue) next(time.Time) (time.Time, error) { func (due scheduleDue) next(time.Time) (time.Time, error) {
return TS(due).time(), nil return TS(due).time(), nil
} }
// 2022-01-01
type scheduleEZDate string
var scheduleEZDatePattern = regexp.MustCompile(`^[0-9]{4}-[0-9]{2}-[0-9]{2}$`)
func (ezdate scheduleEZDate) next(time.Time) (time.Time, error) {
return time.Parse("2006-01-02", string(ezdate))
}

View File

@@ -50,6 +50,7 @@ func TestJSONSchedule(t *testing.T) {
func TestSchedulerFactory(t *testing.T) { func TestSchedulerFactory(t *testing.T) {
start := time.Date(2000, 1, 1, 1, 1, 1, 1, time.UTC) start := time.Date(2000, 1, 1, 1, 1, 1, 1, time.UTC)
someDay := time.Date(2001, 2, 3, 0, 0, 0, 0, time.UTC)
cases := map[string]struct { cases := map[string]struct {
input string input string
want interface{} want interface{}
@@ -70,11 +71,6 @@ func TestSchedulerFactory(t *testing.T) {
want: scheduleDue(1), want: scheduleDue(1),
next: time.Unix(1, 0), next: time.Unix(1, 0),
}, },
"zero ts": {
input: `0`,
want: scheduleDue(0),
next: time.Unix(0, 0),
},
"never": { "never": {
input: ``, input: ``,
want: scheduleNever{}, want: scheduleNever{},
@@ -90,6 +86,11 @@ func TestSchedulerFactory(t *testing.T) {
want: scheduleCron(`5 * * * *`), want: scheduleCron(`5 * * * *`),
next: start.Add(time.Duration(-1*start.Nanosecond() + -1*start.Minute() + -1*start.Second())).Add(5 * time.Minute), next: start.Add(time.Duration(-1*start.Nanosecond() + -1*start.Minute() + -1*start.Second())).Add(5 * time.Minute),
}, },
"ezdate": {
input: `2000-01-03`,
want: scheduleEZDate(`2000-01-03`),
next: someDay,
},
} }
for name, d := range cases { for name, d := range cases {
@@ -108,4 +109,18 @@ func TestSchedulerFactory(t *testing.T) {
} }
}) })
} }
t.Run("zero ts", func(t *testing.T) {
got := schedulerFactory("0")
if fmt.Sprintf("%T", scheduleDue(0)) != fmt.Sprintf("%T", got) {
t.Fatalf("want type %T, got %T", scheduleDue(0), got)
}
if fmt.Sprint(scheduleDue(0)) != fmt.Sprint(got) {
t.Fatalf("want %+v, got %+v", scheduleDue(0), got)
}
next, _ := got.next(start)
if now := time.Now(); next.Sub(now) > time.Second {
t.Fatalf("want next %+v, got %+v", now, next)
}
})
} }

View File

@@ -7,11 +7,12 @@ import (
type Todo struct { type Todo struct {
Todo string Todo string
TS TS
Details string `yaml:",omitempty"` Details string `yaml:",omitempty"`
Schedule Schedule `yaml:",omitempty"` Schedule Schedule `yaml:",omitempty"`
Tags string `yaml:",omitempty"` Tags string `yaml:",omitempty"`
Subtasks []Todo `yaml:",omitempty"` Subtasks []Todo `yaml:",omitempty"`
TS TS `yaml:",omitempty"`
writeTS bool
} }
func (todo Todo) Triggered() bool { func (todo Todo) Triggered() bool {
@@ -21,6 +22,11 @@ func (todo Todo) Triggered() bool {
} }
func (todo Todo) MarshalYAML() (interface{}, error) { func (todo Todo) MarshalYAML() (interface{}, error) {
if !todo.writeTS {
todo.TS = 0
} else {
todo.TS = TS(todo.TS.time().Unix())
}
if fmt.Sprint(todo) == fmt.Sprint(Todo{Todo: todo.Todo}) { if fmt.Sprint(todo) == fmt.Sprint(Todo{Todo: todo.Todo}) {
return todo.Todo, nil return todo.Todo, nil
} }

View File

@@ -11,6 +11,9 @@ import (
type TS int64 type TS int64
func (ts TS) time() time.Time { func (ts TS) time() time.Time {
if ts == 0 {
ts = TS(time.Now().Unix())
}
return time.Unix(int64(ts), 0) return time.Unix(int64(ts), 0)
} }
@@ -26,8 +29,7 @@ func (ts TS) MarshalYAML() (interface{}, error) {
if ts == 0 { if ts == 0 {
ts = TS(time.Now().Unix()) ts = TS(time.Now().Unix())
} }
t := time.Unix(int64(ts), 0) return ts.time().Format(time.UnixDate), nil
return t.Format(time.UnixDate), nil
} }
func (ts *TS) UnmarshalJSON(b []byte) error { func (ts *TS) UnmarshalJSON(b []byte) error {

View File

@@ -1,12 +1,21 @@
todo: todo:
- when merging, check for complete/incomplete and cross reference todo vs scheduled
vs done
- how to defer a scheduled task?
- merge multi todo files for 'inbox' style with some dedupe (probably best to just
do this via calling pttodo-cli from script or somethin) (probably at an off hour
so no collisions from live editing)
- click on links when dumped via cli
- what do about todo-now vs todo vs watch vs later... could do many files and manage
outside binary, but search would suck. Good would be vendor lockin, and that's UI
problem
- todo: when to run scheduled modifier? like, syncthing could have conflicts if I - todo: when to run scheduled modifier? like, syncthing could have conflicts if I
modify only file on remote modify only file on remote
ts: Fri Dec 31 22:33:12 EST 2021
details: | details: |
- if it's a web ui hook or somethin, then it'd only have file conflict if I modify without waiting - if it's a web ui hook or somethin, then it'd only have file conflict if I modify without waiting
- but thats a step back from current todo solution - but thats a step back from current todo solution
tags: stuffToTry,secondTag
- todo: ez edit on many platforms, even offline and mobile - todo: ez edit on many platforms, even offline and mobile
ts: Fri Dec 31 22:33:12 EST 2021
details: | details: |
mobile view + complete method mobile view + complete method
collab editing of file prob resolves mobile and other stuff... collab editing of file prob resolves mobile and other stuff...
@@ -14,35 +23,60 @@ todo:
web server access to ops web server access to ops
is a web ui for mobile best solution? is a web ui for mobile best solution?
let git be smart-ish and keep latest? would provide versioning OOTB without touching raw let git be smart-ish and keep latest? would provide versioning OOTB without touching raw
- xactions emails extend ledger,todo-server with pttodo
- xactions emails extend ledger,todo-server with pttodo - schedule merge inbox style
scheduled: [] scheduled: []
done: done:
- ts to human readable - todo: add tag anti-search, like -tag=-notMe
- here is my really here is my really here is my really here is my really here is ts: Tue Jan 4 06:56:03 EST 2022
my really here is my really here is my really here is my really here is my really - todo: if in todo, then omit ts field
here is my really here is my really here is my really here is my really here is ts: Sun Jan 2 20:44:27 EST 2022
my really here is my really long string - todo: merge full files to import from todo-server
- vim doesnt source vimrc, so stuff like tab width and tab characters, also syntax ts: Sun Jan 2 20:44:27 EST 2022
highlight :( - todo: more schedule formats, like just 2022-01-15, for deferring
- crontab -e style editing to ensure good syntax ts: Sun Jan 2 20:44:27 EST 2022
- todo: from todo-server to pttodo format
ts: Sun Jan 2 20:44:27 EST 2022
- todo: add -tag to search via cli
ts: Sun Jan 2 20:44:27 EST 2022
- todo: ts to human readable
ts: Sun Jan 2 20:44:27 EST 2022
- todo: here is my really here is my really here is my really here is my really here
is my really here is my really here is my really here is my really here is my
really here is my really here is my really here is my really here is my really
here is my really here is my really long string
ts: Sun Jan 2 20:44:27 EST 2022
- todo: vim doesnt source vimrc, so stuff like tab width and tab characters, also
syntax highlight :(
ts: Sun Jan 2 20:44:27 EST 2022
- todo: crontab -e style editing to ensure good syntax
ts: Sun Jan 2 20:44:27 EST 2022
- todo: YAML based todo - todo: YAML based todo
ts: Fri Dec 31 22:33:12 EST 2021
details: because goddamnit a year of this shit isn't significant on disk or in RAM details: because goddamnit a year of this shit isn't significant on disk or in RAM
for vim for vim
ts: Fri Dec 31 22:33:12 EST 2021
- todo: yaml based todo for plaintext - todo: yaml based todo for plaintext
ts: Fri Dec 31 22:33:12 EST 2021
details: a year isnt even a mb details: a year isnt even a mb
- ez edit, start on many platforms
- defer
- schedule
- looping
- details
- let UI be UI for whatever platform
- tags
- todo: sub tasks
ts: Fri Dec 31 22:33:12 EST 2021 ts: Fri Dec 31 22:33:12 EST 2021
- todo: ez edit, start on many platforms
ts: Sun Jan 2 20:44:27 EST 2022
- todo: defer
ts: Sun Jan 2 20:44:27 EST 2022
- todo: schedule
ts: Sun Jan 2 20:44:27 EST 2022
- todo: looping
ts: Sun Jan 2 20:44:27 EST 2022
- todo: details
ts: Sun Jan 2 20:44:27 EST 2022
- todo: let UI be UI for whatever platform
ts: Sun Jan 2 20:44:27 EST 2022
- todo: tags
ts: Sun Jan 2 20:44:27 EST 2022
- todo: sub tasks
subtasks: subtasks:
- a - a
ts: Fri Dec 31 22:33:12 EST 2021
- todo: crap losing on a bad edit hurts - todo: crap losing on a bad edit hurts
ts: Fri Dec 31 22:37:58 EST 2021
details: | details: |
? ?
ts: Fri Dec 31 22:37:58 EST 2021