*/ class TaskManager { /** * @var Taskwarrior */ private $taskwarrior; /** * @var Task[] */ private $tasks = []; /** * @param Taskwarrior $taskwarrior */ public function __construct(Taskwarrior $taskwarrior) { $this->taskwarrior = $taskwarrior; } /** * @return Taskwarrior */ public function getTaskwarrior() { return $this->taskwarrior; } /** * @param Task $task * @throws TaskwarriorException */ public function save(Task $task) { $errors = $this->validate($task); if ($errors) { throw new TaskwarriorException(implode(', ', $errors)); } if (!$task->getUuid()) { $this->add($task); } else { $this->edit($task); } $this->refresh($task); } /** * @param string $uuid * @return Task * @throws TaskwarriorException */ public function find($uuid) { if (isset($this->tasks[$uuid])) { return $this->tasks[$uuid]; } $tasks = $this->filterAll($uuid); if (count($tasks) == 0) { return null; } if (count($tasks) == 1) { return $tasks[0]; } throw new TaskwarriorException(); } /** * @param string $filter * @return Task[]|ArrayCollection */ public function filterAll($filter = null) { $result = $this->export($filter); foreach ($result as $key => $task) { if (isset($this->tasks[$task->getUuid()])) { $result[$key] = $this->tasks[$task->getUuid()]; $this->merge($result[$key], $task); continue; } $this->tasks[$task->getUuid()] = $task; } return new ArrayCollection($result); } /** * @param string|array $filter * @return Task[]|ArrayCollection */ public function filter($filter = null) { return $this->filterAll($filter . ' status:pending'); } /** * @param Task $task */ public function delete(Task $task) { if (!$task->getUuid()) { return; } if ($task->isRecurring()) { $task->setUntil('now'); $this->save($task); } else { $this->taskwarrior->delete($task->getUuid()); $this->refresh($task); } } /** * @param Task $task */ public function done(Task $task) { if (!$task->getUuid()) { return; } $this->taskwarrior->done($task->getUuid()); $this->refresh($task); } /** * @param Task $task */ public function start(Task $task) { if (!$task->getUuid()) { return; } $this->taskwarrior->start($task->getUuid()); $this->refresh($task); } /** * @param Task $task */ public function stop(Task $task) { if (!$task->getUuid()) { return; } $this->taskwarrior->stop($task->getUuid()); $this->refresh($task); } /** * @param Task $task */ public function reopen(Task $task) { if (!$task->getUuid()) { return; } if ($task->isPending() || $task->isWaiting() || $task->isRecurring()) { return; } $this->taskwarrior->modify([ 'status' => Task::STATUS_PENDING ], $task->getUuid()); $this->refresh($task); } /** * @param Task $task * @return array */ public function validate(Task $task) { $errors = []; if ($task->isRecurring() && !$task->getDue()) { $errors[] = 'You cannot remove the due date from a recurring task.'; } if ($task->isRecurring() && !$task->getRecurring()) { $errors[] = 'You cannot remove the recurrence from a recurring task.'; } if ($task->getRecurring() && !$task->getDue()) { $errors[] = "A recurring task must also have a 'due' date."; } return $errors; } /** * @param Task $task */ public function refresh(Task $task) { $clean = $this->export($task->getUuid())[0]; $this->merge($task, $clean); } /** * */ public function clear() { $this->tasks = []; } /** * @return QueryBuilder */ public function createQueryBuilder() { return new QueryBuilder($this); } /** * @param Report|string $report * @return Task[]|ArrayCollection * @throws Exception\ConfigException */ public function filterByReport($report) { if (!$report instanceof Report) { $report = $this->taskwarrior->config()->getReport($report); } return $this->createQueryBuilder() ->where($report->filter) ->orderBy($report->sort) ->getResult(); } /** * @param Context|string $context * @return Task[]|ArrayCollection * @throws Exception\ConfigException */ public function filterByContext($context) { if (!$context instanceof Report) { $context = $this->taskwarrior->config()->getContext($context); } return $this->createQueryBuilder() ->where($context->filter) ->getResult(); } /** * @param string|array $filter * @return Task[] */ private function export($filter = null) { $this->update(); $json = $this->taskwarrior->export($filter); return $this->getSerializer()->deserialize($json, 'array', 'json'); } /** * @param Task $task * @throws TaskwarriorException */ private function add(Task $task) { $json = $this->serializeTask($task); $uuid = $this->taskwarrior->import($json); $this->setValue($task, 'uuid', $uuid); $this->tasks[$uuid] = $task; } /** * @param Task $task * @throws TaskwarriorException */ private function edit(Task $task) { $params = [ 'description' => $task->getDescription(), 'project' => $task->getProject(), 'priority' => $task->getPriority(), 'tags' => $task->getTags(), 'due' => $this->transformDate($task->getDue()), 'wait' => $this->transformDate($task->getWait()), 'until' => $this->transformDate($task->getUntil()) ]; if ($task->getRecurring()) { $params['recur'] = $task->getRecurring()->getValue(); } $this->taskwarrior->modify($params, $task->getUuid()); } /** * @param Task $old * @param Task $new */ private function merge(Task $old, Task $new) { $this->setValue($old, 'urgency', $new->getUrgency()); $this->setValue($old, 'status', $new->getStatus()); $this->setValue($old, 'modified', $new->getModified()); $this->setValue($old, 'start', $new->getStart()); if ($new->isPending()) { // fix reopen problem $this->setValue($old, 'end', null); } else { $this->setValue($old, 'end', $new->getEnd()); } } /** * */ private function update() { try { $this->taskwarrior->command('list'); } catch (TaskwarriorException $e) { // do nothing } } /** * * @param Task $task * @return string */ private function serializeTask(Task $task) { $result = $this->getSerializer()->serialize($task, 'json'); return str_replace("\\/", "/", $result); } /** * @param Task $task * @param string $attr * @param mixed $value */ private function setValue(Task $task, $attr, $value) { $refClass = new \ReflectionClass('DavidBadura\Taskwarrior\Task'); $refProp = $refClass->getProperty($attr); $refProp->setAccessible(true); $refProp->setValue($task, $value); } /** * @param \DateTime $dateTime * @return null|string */ private function transformDate(\DateTime $dateTime = null) { if (!$dateTime) { return null; } $dateTime = clone $dateTime; $dateTime->setTimezone(new \DateTimeZone('UTC')); return $dateTime->format('Ymd\THis\Z'); } /** * @return Serializer */ private function getSerializer() { $propertyNamingStrategy = new SerializedNameAnnotationStrategy(new CamelCaseNamingStrategy()); $visitor = new JsonSerializationVisitor($propertyNamingStrategy); $visitor->setOptions(JSON_UNESCAPED_UNICODE); return SerializerBuilder::create() ->setPropertyNamingStrategy($propertyNamingStrategy) ->configureHandlers(function (HandlerRegistryInterface $registry) { $registry->registerSubscribingHandler(new CarbonHandler()); $registry->registerSubscribingHandler(new RecurringHandler()); }) ->addDefaultHandlers() ->setSerializationVisitor('json', $visitor) ->addDefaultDeserializationVisitors() ->build(); } /** * @return self */ public static function create() { return new self(new Taskwarrior()); } }