initial checkin
This commit is contained in:
commit
089e6d40e6
6 changed files with 232 additions and 0 deletions
39
due-eod.py
Executable file
39
due-eod.py
Executable file
|
@ -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)
|
1
on-add.due-eod
Symbolic link
1
on-add.due-eod
Symbolic link
|
@ -0,0 +1 @@
|
||||||
|
due-eod.py
|
19
on-exit-sync.sh
Executable file
19
on-exit-sync.sh
Executable file
|
@ -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
|
1
on-modify.due-eod
Symbolic link
1
on-modify.due-eod
Symbolic link
|
@ -0,0 +1 @@
|
||||||
|
due-eod.py
|
70
on-modify.jrnl
Executable file
70
on-modify.jrnl
Executable file
|
@ -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')
|
||||||
|
|
102
on-modify.timewarrior
Executable file
102
on-modify.timewarrior
Executable file
|
@ -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])
|
Loading…
Reference in a new issue