call taskwarrior

This commit is contained in:
Johann Dreo 2023-08-10 12:44:05 +02:00
commit 8a90e5b2e4

127
twf.py
View file

@ -1,4 +1,5 @@
import os import os
import re
import sys import sys
import json import json
import argparse import argparse
@ -9,31 +10,16 @@ import rich
from rich.console import Console from rich.console import Console
from rich.columns import Columns from rich.columns import Columns
# class make:
# def carder(card, **kwargs):
# def f(task):
# return card(task, **kwargs)
# return f
# def stacker(stack, **kwargs):
# def f(tasks):
# return stack(tasks, **kwargs)
# return f
# def sectioner(section, **kwargs):
# def f(tasks, field, values):
# return section(tasks, field, values, **kwargs)
# return f
class Widget: class Widget:
def __init__(self, order, group = None): def __init__(self, order, group = None):
self.order = order self.order = order
self.group = group self.group = group
class Tasker(Widget): class Tasker(Widget):
def __init__(self, show_only, order, group = None): def __init__(self, show_only, order, group = None, touched = []):
super().__init__(order, group) super().__init__(order, group)
self.show_only = show_only self.show_only = show_only
self.touched = touched
def __call__(self, task): def __call__(self, task):
raise NotImplementedError raise NotImplementedError
@ -58,8 +44,8 @@ class Sectioner(Widget):
class task: class task:
class Card(Tasker): class Card(Tasker):
def __init__(self, show_only, order = None): def __init__(self, show_only, order = None, touched = []):
super().__init__(show_only, order, group = None) super().__init__(show_only, order, group = None, touched = touched)
def __call__(self, task): def __call__(self, task):
if not self.show_only: if not self.show_only:
@ -74,6 +60,9 @@ class task:
desc = task["description"] desc = task["description"]
title = sid title = sid
if sid in touched:
title = "*"+title+"*"
segments = [] segments = []
for key in self.show_only: for key in self.show_only:
if key in task.keys() and key not in ["id", "description"]: if key in task.keys() and key not in ["id", "description"]:
@ -171,37 +160,12 @@ class group:
return groups return groups
def call_taskwarrior(args:list[str] = ['export']) -> str:
if __name__ == "__main__": # Local file.
parser = argparse.ArgumentParser(
description="XXX",
formatter_class=argparse.ArgumentDefaultsHelpFormatter,
)
main = parser.add_argument_group('Main')
main.add_argument("-s", "--show", metavar="columns", type=str, default="id,urgency,description,tags", nargs=1,
help="Ordered list of columns to show.")
config = parser.add_argument_group('Config')
config.add_argument("-d", "--data", metavar="FILE", type=str, default=".task", nargs=1,
help="The input data file.")
config.add_argument("--list-separator", metavar="CHARACTER", type=str, default=",", nargs=1,
help="Separator used for lists that are passed as options arguments.")
asked = parser.parse_args()
env = os.environ.copy() env = os.environ.copy()
env["TASKDATA"] = asked.data env["TASKDATA"] = asked.data
cmd = ['task', 'export'] cmd = ['task'] + args
showed = asked.show.split(asked.list_separator)
if not showed:
show_only = None
else:
show_only = showed
try: try:
p = subprocess.Popen( " ".join(cmd), p = subprocess.Popen( " ".join(cmd),
stdout=subprocess.PIPE, stdout=subprocess.PIPE,
@ -214,18 +178,67 @@ if __name__ == "__main__":
except subprocess.CalledProcessError as exc: except subprocess.CalledProcessError as exc:
print("ERROR:", exc.returncode, exc.output, err) print("ERROR:", exc.returncode, exc.output, err)
sys.exit(exc.returncode) sys.exit(exc.returncode)
else: else:
return out.decode('utf-8')
def get_data():
out = call_taskwarrior(['export'])
try:
jdata = json.loads(out) jdata = json.loads(out)
print(json.dumps(jdata, indent=4)) except json.decoder.JSONDecodeError as exc:
print("ERROR:", exc.returncode, exc.output)
else:
return jdata
tasker = task.Card(show_only)
stacker = stack.Vertical(tasker)
group_by_status = group.Status()
sort_on_values = sort.OnValues(["pending","started","completed"])
sectioner = sections.Vertical(stacker, sort_on_values, group_by_status)
console = Console() def parse_touched(out):
console.rule("taskwarrior-fancy") return re.findall('Modifying task ([0-9]+)', out)
console.print(sectioner(jdata))
if __name__ == "__main__":
parser = argparse.ArgumentParser(
description="XXX",
formatter_class=argparse.ArgumentDefaultsHelpFormatter,
)
parser.add_argument("-s", "--show", metavar="columns", type=str, default="id,urgency,description,tags", nargs=1,
help="Ordered list of columns to show.")
config = parser.add_argument_group('configuration options')
config.add_argument("-d", "--data", metavar="FILE", type=str, default=".task", nargs=1,
help="The input data file.")
config.add_argument("--list-separator", metavar="CHARACTER", type=str, default=",", nargs=1,
help="Separator used for lists that are passed as options arguments.")
# Capture whatever remains.
parser.add_argument('cmd', nargs="*")
asked = parser.parse_args()
# First pass arguments to taskwarrior and let it do its magic.
out = call_taskwarrior(asked.cmd)
print(out)
touched = parse_touched(out)
# Then get the resulting data.
jdata = get_data()
# print(json.dumps(jdata, indent=4))
showed = asked.show.split(asked.list_separator)
if not showed:
show_only = None
else:
show_only = showed
tasker = task.Card(show_only, touched)
stacker = stack.Vertical(tasker)
group_by_status = group.Status()
sort_on_values = sort.OnValues(["pending","started","completed"])
sectioner = sections.Vertical(stacker, sort_on_values, group_by_status)
console = Console()
console.rule("taskwarrior-fancy")
console.print(sectioner(jdata))