call taskwarrior
This commit is contained in:
parent
52c3622a20
commit
8a90e5b2e4
1 changed files with 63 additions and 50 deletions
127
twf.py
127
twf.py
|
|
@ -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))
|
||||||
|
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue