Files
opalfiles/.local/bin/todo

398 lines
13 KiB
Python
Executable File

#!/usr/bin/env python3
"""
todo.py
A minimalist CLI todo application with priorities, tags, sorting, JIRA import,
and completion-date tracking. Uses INI config at ~/.config/todo/config.ini.
"""
import os
import argparse
import re
import configparser
import urllib.request
import urllib.parse
import urllib.error
import json
from datetime import datetime, date
from pathlib import Path
from rich.console import Console
from rich.table import Table
from rich.prompt import Prompt, IntPrompt
from rich.text import Text
# -------------------------------------------------------------------
# Configuration loading via configparser
# -------------------------------------------------------------------
DEFAULT_CONFIG = {
'todo': {
'todo_file': '~/sync/todo/todo.txt',
'archive_file': '~/sync/todo/todo-archive.txt',
'default_priority': 'B'
},
'tag_colors': {
'@jira': 'blue',
'@me': 'magenta',
'@work': 'green'
},
'priority_labels': {
'A': 'High',
'B': 'Med',
'C': 'Low'
},
'priority_styles': {
'A': 'bold red',
'B': 'yellow',
'C': 'green'
}
}
def load_config():
cfg = configparser.ConfigParser()
cfg.optionxform = str
config_path = Path.home() / '.config' / 'todo' / 'config.ini'
for section, opts in DEFAULT_CONFIG.items():
if not cfg.has_section(section):
cfg[section] = {}
for k, v in opts.items():
cfg[section].setdefault(k, str(v))
if config_path.exists():
cfg.read(config_path)
return {
'todo_file': Path(cfg['todo']['todo_file']).expanduser(),
'archive_file': Path(cfg['todo']['archive_file']).expanduser(),
'default_priority': cfg['todo']['default_priority'],
'tag_colors': dict(cfg['tag_colors']),
'priority_labels': dict(cfg['priority_labels']),
'priority_styles': dict(cfg['priority_styles']),
}
CFG = load_config()
TODO_FILE_PATH = CFG['todo_file']
ARCHIVE_FILE_PATH = CFG['archive_file']
DEFAULT_PRIORITY = CFG['default_priority']
TAG_COLOR_MAP = CFG['tag_colors']
PRIORITY_LABELS = CFG['priority_labels']
PRIORITY_STYLE_MAP= CFG['priority_styles']
DATE_FORMAT = '%Y-%m-%d'
console = Console()
# -------------------------------------------------------------------
# Task class with completion-date support
# -------------------------------------------------------------------
LINE_RE = re.compile(
r'^(?:x \((?P<completed>\d{4}-\d{2}-\d{2})\)\s*)?'
r'(?:\[#(?P<priority>[ABC])\]\s*)?'
r'\((?P<created>\d{4}-\d{2}-\d{2})\)\s*'
r'(?P<description>.+)$'
)
class Task:
def __init__(self, completed, priority, created, description, tags):
self.completed = completed
self.priority = priority
self.created = created
self.description = description
self.tags = tags
@property
def done(self):
return self.completed is not None
@classmethod
def from_line(cls, line):
m = LINE_RE.match(line)
if not m:
return None
gd = m.groupdict()
completed = None
if gd.get('completed'):
completed = datetime.strptime(gd['completed'], DATE_FORMAT).date()
prio = gd.get('priority') or DEFAULT_PRIORITY
created = datetime.strptime(gd['created'], DATE_FORMAT).date()
parts = gd['description'].split()
tags = [w for w in parts if w.startswith('@')]
desc = ' '.join(w for w in parts if not w.startswith('@'))
return cls(completed, prio, created, desc, tags)
def to_line(self):
parts = []
if self.done:
parts.append(f"x ({self.completed.strftime(DATE_FORMAT)}) ")
parts.append(f"[#{self.priority}] ")
parts.append(f"({self.created.strftime(DATE_FORMAT)}) ")
parts.append(self.description)
if self.tags:
parts.append(' ' + ' '.join(self.tags))
return ''.join(parts)
def mark_done(self):
return Task(date.today(), self.priority, self.created, self.description, self.tags)
# -------------------------------------------------------------------
# I/O
# -------------------------------------------------------------------
def read_tasks():
if not TODO_FILE_PATH.exists():
return []
lines = TODO_FILE_PATH.read_text().splitlines()
tasks = []
for l in lines:
t = Task.from_line(l)
if t:
tasks.append(t)
return tasks
def write_tasks(tasks):
TODO_FILE_PATH.parent.mkdir(parents=True, exist_ok=True)
lines = [t.to_line() for t in tasks]
TODO_FILE_PATH.write_text("\n".join(lines) + "\n")
# -------------------------------------------------------------------
# Display & Sorting
# -------------------------------------------------------------------
def display_tasks(tasks, show_all, sort_by):
display = tasks if show_all else [t for t in tasks if not t.done]
if sort_by == 'created':
display.sort(key=lambda t: t.created)
elif sort_by == 'priority':
display.sort(key=lambda t: t.priority)
elif sort_by and sort_by.startswith('tag:'):
tag = sort_by.split(':',1)[1]
display.sort(key=lambda t: tag in t.tags, reverse=True)
cols = ['#','Created','Pri','Description','Tags']
if show_all:
cols.insert(2, 'Completed')
cols.insert(4, 'Done')
table = Table(title='TODOs (all)' if show_all else 'TODOs')
for c in cols:
table.add_column(c, justify='right' if c=='#' else None)
for idx, t in enumerate(display, start=1):
pri_lbl = PRIORITY_LABELS.get(t.priority, t.priority)
pri_cell = Text(pri_lbl, style=PRIORITY_STYLE_MAP.get(t.priority,''))
tag_cell = Text()
for tag in t.tags:
tag_cell.append(tag, TAG_COLOR_MAP.get(tag,''))
tag_cell.append(' ')
row = [str(idx), t.created.strftime(DATE_FORMAT)]
if show_all:
comp = t.completed.strftime(DATE_FORMAT) if t.completed else ''
row.append(comp)
row.append(pri_cell)
if show_all:
row.append('' if t.done else '')
row.append(t.description)
row.append(tag_cell)
table.add_row(*row)
console.print(table)
# -------------------------------------------------------------------
# Commands
# -------------------------------------------------------------------
def cmd_ls(args):
display_tasks(read_tasks(), False, args.sort)
def cmd_lsa(args):
display_tasks(read_tasks(), True, args.sort)
def cmd_add(args):
if args.text:
entry = ' '.join(args.text)
if not entry.startswith('[#'):
console.print('[yellow]No priority; defaulting to Med (B).[/]')
entry = '[#B] ' + entry
if '@' not in entry:
console.print('[yellow]No tag; defaulting to @misc.[/]')
entry += ' @misc'
t = Task.from_line(entry)
tasks = read_tasks()
if t:
tasks.append(t)
write_tasks(tasks)
console.print(f'[green]Added:[/] {t.description}')
else:
desc = Prompt.ask('Task description')
prio = Prompt.ask('Priority (A=High,B=Med,C=Low)', choices=['A','B','C'], default=DEFAULT_PRIORITY)
tag = Prompt.ask('Tag (e.g. @work,@me,@jira)', default='@misc')
t = Task(None, prio, date.today(), desc, [tag])
tasks = read_tasks()
tasks.append(t)
write_tasks(tasks)
console.print(f'[green]Added:[/] {desc}')
def cmd_done(args):
all_tasks = read_tasks()
active_idxs = [i for i,t in enumerate(all_tasks) if not t.done]
try:
n = int(args.number)
if not (1 <= n <= len(active_idxs)):
raise IndexError
idx = active_idxs[n-1]
all_tasks[idx] = all_tasks[idx].mark_done()
write_tasks(all_tasks)
console.print(f"[green]Completed:[/] {all_tasks[idx].description}")
except:
console.print('[red]Invalid task number[/]')
def cmd_donei(args):
all_tasks = read_tasks()
active = [(i,t) for i,t in enumerate(all_tasks) if not t.done]
if not active:
console.print('[yellow]No active tasks.[/]')
return
for disp,(i,t) in enumerate(active, start=1):
console.print(f"{disp}: {t.description}")
choice = IntPrompt.ask('Complete task #')
cmd_done(argparse.Namespace(number=str(choice)))
def cmd_archive(args):
tasks = read_tasks()
done = [t for t in tasks if t.done]
if not done:
console.print('[yellow]No tasks to archive[/]')
return
ARCHIVE_FILE_PATH.parent.mkdir(parents=True, exist_ok=True)
with ARCHIVE_FILE_PATH.open('a') as f:
for t in done:
f.write(t.to_line() + "\n")
remaining = [t for t in tasks if not t.done]
write_tasks(remaining)
console.print(f'[green]Archived {len(done)}[/]')
def cmd_tags(args):
tasks = [t for t in read_tasks() if args.tag in t.tags and not t.done]
if not tasks:
console.print(f'[yellow]No active tasks with tag {args.tag}[/]')
else:
display_tasks(tasks, False, None)
def cmd_edit(args):
editor = os.getenv('EDITOR','nano')
os.execvp(editor, [editor, str(TODO_FILE_PATH)])
def cmd_jira(args):
jira_token = os.getenv('JIRA_TOKEN')
if not jira_token:
console.print('[red]Missing JIRA_TOKEN environment variable[/]')
return
base_url = 'https://jira.atg-corp.com'
jql = (
f'project="{args.project}" '
f'AND assignee=currentUser() '
f'AND status in ("Open","In Progress") '
f'ORDER BY created DESC'
)
url = f'{base_url}/rest/api/2/search?jql={urllib.parse.quote(jql)}'
console.print(f'[cyan]Requesting URL:[/] {url}')
req = urllib.request.Request(url)
req.add_header('Authorization', f'Bearer {jira_token}')
req.add_header('Accept', 'application/json')
req.add_header('Content-Type', 'application/json')
try:
with urllib.request.urlopen(req) as resp:
data = json.load(resp)
except urllib.error.HTTPError as e:
body = e.read().decode(errors='ignore')
console.print(f'[red]HTTP Error {e.code}: {e.reason}[/]')
console.print('[yellow]Response body:[/]')
console.print(body)
return
except Exception as e:
console.print(f'[red]Failed to fetch JIRA issues:[/] {e}')
return
issues = data.get('issues', [])
if not issues:
console.print('[yellow]No JIRA issues found.[/]')
return
tasks = read_tasks()
added = 0
for issue in issues:
key = issue['key']
summary = issue['fields']['summary']
date_str = issue['fields']['created'][:10]
desc = f'[{key}] {summary}'
# skip duplicates by key
if any(t.description.startswith(f'[{key}]') for t in tasks):
continue
t = Task(
None,
'B',
datetime.strptime(date_str, DATE_FORMAT).date(),
desc,
['@jira']
)
tasks.append(t)
added += 1
write_tasks(tasks)
console.print(f'[green]Imported {added} JIRA tasks from {args.project} assigned to you.[/]')
# -------------------------------------------------------------------
# CLI dispatch
# -------------------------------------------------------------------
def main():
parser = argparse.ArgumentParser(prog='todo')
sub = parser.add_subparsers(dest='cmd', required=True)
p_ls = sub.add_parser('ls', help='List active tasks')
p_ls.add_argument('--sort', choices=['created','priority','tag:@jira','tag:@work','tag:@me'])
p_lsa = sub.add_parser('lsa', help='List all tasks')
p_lsa.add_argument('--sort', choices=['created','priority','tag:@jira','tag:@work','tag:@me'])
sub.add_parser('archive', help='Archive completed tasks')
p_add = sub.add_parser('add', help='Add a task')
p_add.add_argument('text', nargs=argparse.REMAINDER)
p_done = sub.add_parser('done', help='Mark a task done')
p_done.add_argument('number')
sub.add_parser('donei', help='Interactive completion')
p_tags = sub.add_parser('tags', help='Filter by tag')
p_tags.add_argument('tag')
sub.add_parser('edit', help='Open todo file in editor')
p_jira = sub.add_parser('jira', help='Import JIRA issues')
p_jira.add_argument('--project', default='IS', help='JIRA project key (defaults to IS)')
args = parser.parse_args()
{
'ls': cmd_ls,
'lsa': cmd_lsa,
'add': cmd_add,
'done': cmd_done,
'donei': cmd_donei,
'archive': cmd_archive,
'tags': cmd_tags,
'edit': cmd_edit,
'jira': cmd_jira,
}[args.cmd](args)
if __name__ == '__main__':
try:
main()
except KeyboardInterrupt:
console.print('\n[red]^C Aborted[/]')