Source code for taskw.task

import json
import logging
import os
import sys

import six

from taskw.fields import (
    AnnotationArrayField,
    ArrayField,
    ChoiceField,
    CommaSeparatedUUIDField,
    DateField,
    DurationField,
    Field,
    NumericField,
    StringField,
    UUIDField,
)
from taskw.fields.base import Dirtyable, DirtyableList, DirtyableDict

# Sentinel value for not specifying a default
UNSPECIFIED = object()


logger = logging.getLogger(__name__)


[docs]class Task(dict): FIELDS = { 'annotations': AnnotationArrayField(label='Annotations'), 'depends': CommaSeparatedUUIDField(label='Depends Upon'), 'description': StringField(label='Description'), 'due': DateField(label='Due'), 'end': DateField(label='Ended'), 'entry': DateField(label='Entered'), 'id': NumericField(label='ID', read_only=True), 'imask': NumericField(label='Imask', read_only=True), 'mask': StringField(label='Mask', read_only=True), 'modified': DateField(label='Modified'), 'parent': StringField(label='Parent'), 'priority': ChoiceField( choices=[None, 'H', 'M', 'L', ], case_sensitive=False, label='Priority' ), 'project': StringField(label='Project'), 'recur': DurationField(label='Recurrence'), 'scheduled': DateField(label='Scheduled'), 'start': DateField(label='Started'), 'status': ChoiceField( choices=[ 'pending', 'completed', 'deleted', 'waiting', 'recurring', ], case_sensitive=False, label='Status', ), 'tags': ArrayField(label='Tags'), 'until': DateField(label='Until'), 'urgency': NumericField(label='Urgency', read_only=True), 'uuid': UUIDField(label='UUID'), 'wait': DateField(label='Wait'), } def __init__(self, data, udas=None): udas = udas or {} self._fields = self.FIELDS.copy() self._fields.update(udas) self._changes = [] processed = {} for k, v in six.iteritems(data): processed[k] = self._deserialize(k, v, self._fields) super(Task, self).__init__(processed)
[docs] @classmethod def from_stub(cls, data, udas=None): """ Create a Task from an already deserialized dict. """ udas = udas or {} fields = cls.FIELDS.copy() fields.update(udas) processed = {} for k, v in six.iteritems(data): processed[k] = cls._serialize(k, v, fields) return cls(processed, udas)
[docs] @classmethod def from_input(cls, input_file=sys.stdin, modify=False, udas=None): """ Create a Task directly from stdin by reading one line. If modify=True, two lines are expected, which is consistent with the Taskwarrior hook system. The first line is interpreted as the original state of the Task, and the second one as the new, modified state. :param input_file: Input file. Defaults to sys.stdin. :param modify: Flag for on-modify hook event. Defaults to False. :param udas: Taskrc udas. Defaults to None. :return Task """ original_task = input_file.readline().strip() if modify: modified_task = input_file.readline().strip() return cls(json.loads(modified_task), udas=udas) return cls(json.loads(original_task), udas=udas)
@classmethod def _get_converter_for_field(cls, field, default=None, fields=None): fields = fields or {} converter = fields.get(field, None) if not converter: return default if default else Field() return converter @classmethod def _deserialize(cls, key, value, fields): """ Marshal incoming data into Python objects.""" converter = cls._get_converter_for_field(key, None, fields) return converter.deserialize(value) @classmethod def _serialize(cls, key, value, fields): """ Marshal outgoing data into Taskwarrior's JSON format.""" converter = cls._get_converter_for_field(key, None, fields) return converter.serialize(value) def _field_is_writable(self, key): converter = self._get_converter_for_field(key, fields=self._fields) if converter.read_only: return False return True
[docs] def get(self, key, default=None): try: return self[key] except KeyError: return default
def _record_change(self, key, from_, to): self._changes.append((key, from_, to))
[docs] def get_changes(self, serialized=False, keep=False): """ Get a journal of changes that have occurred :param `serialized`: Return changes in the serialized format used by TaskWarrior. :param `keep_changes`: By default, the list of changes is reset after running ``.get_changes``; set this to `True` if you would like to keep the changes recorded following running this command. :returns: A dictionary of 2-tuples of changes, where the key is the name of the field that has changed, and the value is a 2-tuple containing the original value and the final value respectively. """ results = {} # Check for explicitly-registered changes for k, f, t in self._changes: if k not in results: results[k] = [f, None] results[k][1] = ( self._serialize(k, t, self._fields) if serialized else t ) # Check for changes on subordinate items for k, v in six.iteritems(self): if isinstance(v, Dirtyable): result = v.get_changes(keep=keep) if result: if not k in results: results[k] = [result[0], None] results[k][1] = ( self._serialize(k, result[1], self._fields) if serialized else result[1] ) # Clear out recorded changes if not keep: self._changes = [] return results
[docs] def update(self, values, force=False): """ Update this task dictionary :returns: A dictionary mapping field names specified to be updated and a boolean value indicating whether the field was changed. """ results = {} for k, v in six.iteritems(values): results[k] = self.__setitem__(k, v, force=force) return results
[docs] def set(self, key, value): """ Set a key's value regardless of whether a change is seen.""" return self.__setitem__(key, value, force=True)
[docs] def serialized(self): """ Returns a serialized representation of this task.""" serialized = {} for k, v in six.iteritems(self): serialized[k] = self._serialize(k, v, self._fields) return serialized
[docs] def serialized_changes(self, keep=False): serialized = {} for k, v in six.iteritems(self.get_changes(keep=keep)): # Here, `v` is a 2-tuple of the field's original value # and the field's new value. _, to = v serialized[k] = self._serialize(k, to, self._fields) return serialized
def __setitem__(self, key, value, force=False): if isinstance(value, dict) and not isinstance(value, DirtyableDict): value = DirtyableDict(value) elif isinstance(value, list) and not isinstance(value, DirtyableList): value = DirtyableList(value) existing_value = self.get(key) if force or value != existing_value: if force or existing_value or value: # Do not attempt to record changes if both the existing # and previous values are Falsy. We cannot distinguish # between `''` and `None` for...reasons. self._record_change( key, self.get(key), value, ) # Serialize just to make sure we can; it's better to throw # this error early. self._serialize(key, value, self._fields) # Also make sure we raise an error if this field isn't # writable at all. if not self._field_is_writable(key): raise ValueError("%s is a read-only field", key) super(Task, self).__setitem__(key, value) return True return False