#!/usr/bin/env python3 ## Todoist CLI - command line client for todoist.com online task manager ## ## Copyright (C) 2016 Robin Obůrka ## ## This program is free software: you can redistribute it and/or modify ## it under the terms of the GNU General Public License as published by ## the Free Software Foundation, either version 3 of the License, or ## (at your option) any later version. ## ## This program is distributed in the hope that it will be useful, ## but WITHOUT ANY WARRANTY; without even the implied warranty of ## MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the ## GNU General Public License for more details. ## ## You should have received a copy of the GNU General Public License ## along with this program. If not, see . import os import sys import argparse import json import todoist from pprint import pprint TOKEN_FILE_NAME = ".todoist_token" DUMP_FILENAME = "dump.txt" def arg_parser(): parser = argparse.ArgumentParser() parser.add_argument("--token", metavar="ACCESS_TOKEN_STRING", help="Store or change access token", type=str) parser.add_argument("--get-token", help="Get current token (e.g. for revocation)", action="store_true") parser.add_argument("--dump", help="Dump data to file to " + DUMP_FILENAME + " and exit", action="store_true") parser.add_argument("-p", "--project", metavar="PROJECT_NAME", help="Project name (lowercase!)", type=str) parser.add_argument("-s", "--severity", help="Severity (or priority) of task", type=int, choices=[1, 2, 3, 4]) parser.add_argument("-a", "--assign", metavar="NAME", help="Assign to collaborator", type=str) parser.add_argument("-d", "--date", metavar="DATE", help="Date string", type=str, default="today") parser.add_argument("-n", "--no-due-date", help="Add task with no due date", action="store_true") parser.add_argument("msg", metavar="MSG", nargs='*', help="Content of new Task - stdin if absent") return parser.parse_args() def token_path(): return os.path.join(os.path.expanduser("~"), TOKEN_FILE_NAME) def get_token(): path = token_path() with open(path, mode="r") as f: for line in f: token = line.strip() ## Get first line only break return token def store_token(token): path = token_path() with open(path, mode="w+") as f: f.write(token) def build_project_list(data): projects = { proj["name"].lower(): proj["id"] for proj in data["Projects"] } return projects def build_collaborators_list(data): collaborators = { col["full_name"].lower(): col["id"] for col in data["Collaborators"] } return collaborators def assign_to_person(name, collaborators): candidates = [] for k, v in collaborators.items(): if k.find(name) != -1: candidates.append((k, v)) if len(candidates) == 0: return None, "No candidate found" elif len(candidates) > 1: clist = [ item[0] for item in candidates ] clist.insert(0, "Multiple candidates found. Make better specification of one of them:") return None, "\n".join(clist) else: return candidates[0][1], None def main(): args = arg_parser() if args.token: store_token(args.token) print("Token stored") sys.exit(0) token = get_token() if not token: print("Couldn't read token. Is token stored?", file=sys.stderr) sys.exit(1) if args.get_token: print(token) sys.exit(0) api = todoist.TodoistAPI(token) data = api.sync(resource_types=['all']) if "error" in data: print("Failed to connect or authenticate to server:", file=sys.stderr) print(data["error"], file=sys.stderr) sys.exit(1) if args.dump: with open(DUMP_FILENAME, mode="w") as f: pprint(data, f) sys.exit(0) projects = build_project_list(data) collaborators = build_collaborators_list(data) ## Prepare data if not args.msg: msg = sys.stdin.read().strip() else: msg = " ".join(args.msg) ## Prepare arguments kwargs = {} ## Set date if not args.no_due_date: kwargs["date_string"] = args.date ## Set project if args.project: proj = args.project.lower() if proj in projects: kwargs["project_id"] = projects[proj] else: print("WARNING: Unknown project. Task will be stored in Inbox", file=sys.stderr) ## Set priority/severity if args.severity: kwargs["priority"] = args.severity ## Assign to person if args.assign: aid, err = assign_to_person(args.assign.lower(), collaborators) if not aid: print(err, file=sys.stderr) sys.exit(1) kwargs["responsible_uid"] = aid ## Store task status = api.add_item(msg, **kwargs) if "error" in status: print("Store failed:", file=sys.stderr) print(status["error"], file=sys.stderr) sys.exit(1) api.commit() if __name__ == "__main__": main()