From 089e6d40e64924eeec8a44b4024865243914aa4a Mon Sep 17 00:00:00 2001 From: Alex Kelly Date: Thu, 26 Jan 2023 13:34:13 -0500 Subject: [PATCH] initial checkin --- due-eod.py | 39 ++++++++++++++++ on-add.due-eod | 1 + on-exit-sync.sh | 19 ++++++++ on-modify.due-eod | 1 + on-modify.jrnl | 70 +++++++++++++++++++++++++++++ on-modify.timewarrior | 102 ++++++++++++++++++++++++++++++++++++++++++ 6 files changed, 232 insertions(+) create mode 100755 due-eod.py create mode 120000 on-add.due-eod create mode 100755 on-exit-sync.sh create mode 120000 on-modify.due-eod create mode 100755 on-modify.jrnl create mode 100755 on-modify.timewarrior diff --git a/due-eod.py b/due-eod.py new file mode 100755 index 0000000..9fd42e6 --- /dev/null +++ b/due-eod.py @@ -0,0 +1,39 @@ +#!/usr/bin/env python3 + +# TaskWarrior's hook that sets due date to the end of date. +# By default TaskWarrior sets it to 00:00:00. + +import datetime +import json +import sys + + +def get_task_data() -> str: + """ + Gets task data from stdin. + + It might be first line on task addition and second line on task modification. + + :return: str + """ + input_data = sys.stdin.readlines() + + with open("/tmp/abracadabra", "w") as f: + f.write(input_data[-1]) + + return input_data[-1] + + +task_data_raw = get_task_data() + +task_data = json.loads(task_data_raw) + +if "due" in task_data: + new_due_ts = datetime.datetime.strptime(task_data["due"], "%Y%m%dT%H%M%S%z").astimezone() + if new_due_ts.hour == 00 and new_due_ts.minute == 00 and new_due_ts.second == 00: + new_due_ts = new_due_ts.replace(hour = 23, minute = 59, second = 59) + + task_data["due"] = new_due_ts.astimezone(tz = datetime.timezone.utc).strftime("%Y%m%dT%H%M%SZ") + +print(json.dumps(task_data, ensure_ascii = False)) +sys.exit(0) diff --git a/on-add.due-eod b/on-add.due-eod new file mode 120000 index 0000000..f404e3d --- /dev/null +++ b/on-add.due-eod @@ -0,0 +1 @@ +due-eod.py \ No newline at end of file diff --git a/on-exit-sync.sh b/on-exit-sync.sh new file mode 100755 index 0000000..38c2b40 --- /dev/null +++ b/on-exit-sync.sh @@ -0,0 +1,19 @@ +#!/bin/sh +# This hooks script syncs task warrior to the configured task server. +# The on-exit event is triggered once, after all processing is complete. + +# Make sure hooks are enabled + + +# Count the number of tasks modified +n=0 +while read modified_task +do + n=$(($n + 1)) +done + +if (($n > 0)); then + task sync >> ~/.config/task/sync_hook.log +fi + +exit 0 diff --git a/on-modify.due-eod b/on-modify.due-eod new file mode 120000 index 0000000..f404e3d --- /dev/null +++ b/on-modify.due-eod @@ -0,0 +1 @@ +due-eod.py \ No newline at end of file diff --git a/on-modify.jrnl b/on-modify.jrnl new file mode 100755 index 0000000..71b1929 --- /dev/null +++ b/on-modify.jrnl @@ -0,0 +1,70 @@ +#!/usr/bin/env python +############################################################################### +# +# Copyright 2015 - 2016, Paul Beckingham, Federico Hernandez. +# +# Permission is hereby granted, free of charge, to any person obtaining a copy +# of this software and associated documentation files (the "Software"), to deal +# in the Software without restriction, including without limitation the rights +# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +# copies of the Software, and to permit persons to whom the Software is +# furnished to do so, subject to the following conditions: +# +# The above copyright notice and this permission notice shall be included +# in all copies or substantial portions of the Software. +# +# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS +# OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL +# THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +# SOFTWARE. +# +# http://www.opensource.org/licenses/mit-license.php +# +############################################################################### + +import sys +import json +import os + +# Hook should extract all of the following for use as Timewarrior tags: +# UUID +# Project +# Tags +# Description +# UDAs + +# Make no changes to the task, simply observe. +old = json.loads(sys.stdin.readline()) +new = json.loads(sys.stdin.readline()) +print(json.dumps(new)) + +# Extract attributes for use as tags. +tags = [new['description']] + +if 'project' in new: + project = new['project'] + tags.append(project) + if '.' in project: + tags.extend([tag for tag in project.split('.')]) + +if 'tags' in new: + tags.extend(new['tags']) + +combined = ' +'.join(['"%s"' % tag for tag in tags]).encode('utf-8').strip() +# Started task. +if 'start' in new and not 'start' in old: + os.system('echo Started: ' + combined.decode() + ' @task|jj') + +elif new['status'] == 'completed': + os.system('echo Completed: ' + combined.decode() + ' @task|jj') +# Stopped task. +elif not 'start' in new and 'start' in old: + os.system('echo Stopped: ' + combined.decode() + ' @task|jj') + +# Any task that is active, with a non-pending status should not be tracked. +elif 'start' in new and new['status'] != 'pending': + os.system('echo ' + combined.decode() + ' @task|jj') + diff --git a/on-modify.timewarrior b/on-modify.timewarrior new file mode 100755 index 0000000..46d82a9 --- /dev/null +++ b/on-modify.timewarrior @@ -0,0 +1,102 @@ +#!/usr/bin/env python3 + +############################################################################### +# +# Copyright 2016 - 2021, Thomas Lauf, Paul Beckingham, Federico Hernandez. +# +# Permission is hereby granted, free of charge, to any person obtaining a copy +# of this software and associated documentation files (the "Software"), to deal +# in the Software without restriction, including without limitation the rights +# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +# copies of the Software, and to permit persons to whom the Software is +# furnished to do so, subject to the following conditions: +# +# The above copyright notice and this permission notice shall be included +# in all copies or substantial portions of the Software. +# +# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS +# OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL +# THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +# SOFTWARE. +# +# https://www.opensource.org/licenses/mit-license.php +# +############################################################################### + +from __future__ import print_function + +import json +import subprocess +import sys + +# Hook should extract all of the following for use as Timewarrior tags: +# UUID +# Project +# Tags +# Description +# UDAs + +try: + input_stream = sys.stdin.buffer +except AttributeError: + input_stream = sys.stdin + +# Make no changes to the task, simply observe. +old = json.loads(input_stream.readline().decode("utf-8", errors="replace")) +new = json.loads(input_stream.readline().decode("utf-8", errors="replace")) +print(json.dumps(new)) + + +def extract_tags_from(json_obj): + # Extract attributes for use as tags. + tags = [json_obj['description']] + + if 'project' in json_obj: + tags.append(json_obj['project']) + + if 'tags' in json_obj: + tags.extend(json_obj['tags']) + + return tags + + +def extract_annotation_from(json_obj): + + if 'annotations' not in json_obj: + return '\'\'' + + return json_obj['annotations'][0]['description'] + + +start_or_stop = '' + +# Started task. +if 'start' in new and 'start' not in old: + start_or_stop = 'start' + +# Stopped task. +elif ('start' not in new or 'end' in new) and 'start' in old: + start_or_stop = 'stop' + +if start_or_stop: + tags = extract_tags_from(new) + + subprocess.call(['timew', start_or_stop] + tags + [':yes']) + +# Modifications to task other than start/stop +elif 'start' in new and 'start' in old: + old_tags = extract_tags_from(old) + new_tags = extract_tags_from(new) + + if old_tags != new_tags: + subprocess.call(['timew', 'untag', '@1'] + old_tags + [':yes']) + subprocess.call(['timew', 'tag', '@1'] + new_tags + [':yes']) + + old_annotation = extract_annotation_from(old) + new_annotation = extract_annotation_from(new) + + if old_annotation != new_annotation: + subprocess.call(['timew', 'annotate', '@1', new_annotation])