#!/usr/bin/env python3
###############################################################################
#
# Copyright 2006 - 2021, Tomas Babej, 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
#
###############################################################################

import sys
import os
import unittest
import datetime

# Ensure python finds the local simpletap module
sys.path.append(os.path.dirname(os.path.abspath(__file__)))

from basetest import Task, TestCase


class TestFilter(TestCase):
    @classmethod
    def setUpClass(cls):
        cls.t = Task()

        cls.t("add project:A prio:H +tag one foo")
        cls.t("add project:A prio:H      two")
        cls.t("add project:A             three")
        cls.t("add           prio:H      four")
        cls.t("add                  +tag five")
        cls.t("add                       six foo")
        cls.t("add           prio:L      seven bar foo")

    def test_list(self):
        """filter - list"""
        code, out, err = self.t("list")

        self.assertIn("one", out)
        self.assertIn("two", out)
        self.assertIn("three", out)
        self.assertIn("four", out)
        self.assertIn("five", out)
        self.assertIn("six", out)
        self.assertIn("seven", out)

    def test_list_projectA(self):
        """filter - list project:A"""
        code, out, err = self.t("list project:A")

        self.assertIn("one", out)
        self.assertIn("two", out)
        self.assertIn("three", out)
        self.assertNotIn("four", out)
        self.assertNotIn("five", out)
        self.assertNotIn("six", out)
        self.assertNotIn("seven", out)

    def test_list_priorityH(self):
        """filter - list priority:H"""
        code, out, err = self.t("list priority:H")

        self.assertIn("one", out)
        self.assertIn("two", out)
        self.assertNotIn("three", out)
        self.assertIn("four", out)
        self.assertNotIn("five", out)
        self.assertNotIn("six", out)
        self.assertNotIn("seven", out)

    def test_list_priority(self):
        """filter - list priority:"""
        code, out, err = self.t("list priority:")

        self.assertNotIn("one", out)
        self.assertNotIn("two", out)
        self.assertIn("three", out)
        self.assertNotIn("four", out)
        self.assertIn("five", out)
        self.assertIn("six", out)
        self.assertNotIn("seven", out)

    def test_list_substring(self):
        """filter - list /foo/"""
        code, out, err = self.t("list /foo/")

        self.assertIn("one", out)
        self.assertNotIn("two", out)
        self.assertNotIn("three", out)
        self.assertNotIn("four", out)
        self.assertNotIn("five", out)
        self.assertIn("six", out)
        self.assertIn("seven", out)

    def test_list_double_substring(self):
        """filter - list /foo/ /bar/"""
        code, out, err = self.t("list /foo/ /bar/")

        self.assertNotIn("one", out)
        self.assertNotIn("two", out)
        self.assertNotIn("three", out)
        self.assertNotIn("four", out)
        self.assertNotIn("five", out)
        self.assertNotIn("six", out)
        self.assertIn("seven", out)

    def test_list_include_tag(self):
        """filter - list +tag"""
        code, out, err = self.t("list +tag")

        self.assertIn("one", out)
        self.assertNotIn("two", out)
        self.assertNotIn("three", out)
        self.assertNotIn("four", out)
        self.assertIn("five", out)
        self.assertNotIn("six", out)
        self.assertNotIn("seven", out)

    def test_list_exclude_tag(self):
        """filter - list -tag"""
        code, out, err = self.t("list -tag")

        self.assertNotIn("one", out)
        self.assertIn("two", out)
        self.assertIn("three", out)
        self.assertIn("four", out)
        self.assertNotIn("five", out)
        self.assertIn("six", out)
        self.assertIn("seven", out)

    def test_list_non_existing_tag(self):
        """filter - list -missing"""
        code, out, err = self.t("list -missing")

        self.assertIn("one", out)
        self.assertIn("two", out)
        self.assertIn("three", out)
        self.assertIn("four", out)
        self.assertIn("five", out)
        self.assertIn("six", out)
        self.assertIn("seven", out)

    def test_list_mutually_exclusive_tag(self):
        """filter - list +tag -tag"""
        code, out, err = self.t.runError("list +tag -tag")

        self.assertNotIn("one", out)
        self.assertNotIn("two", out)
        self.assertNotIn("three", out)
        self.assertNotIn("four", out)
        self.assertNotIn("five", out)
        self.assertNotIn("six", out)
        self.assertNotIn("seven", out)

    def test_list_projectA_priorityH(self):
        """filter - list project:A priority:H"""
        code, out, err = self.t("list project:A priority:H")

        self.assertIn("one", out)
        self.assertIn("two", out)
        self.assertNotIn("three", out)
        self.assertNotIn("four", out)
        self.assertNotIn("five", out)
        self.assertNotIn("six", out)
        self.assertNotIn("seven", out)

    def test_list_projectA_priority(self):
        """filter - list project:A priority:"""
        code, out, err = self.t("list project:A priority:")

        self.assertNotIn("one", out)
        self.assertNotIn("two", out)
        self.assertIn("three", out)
        self.assertNotIn("four", out)
        self.assertNotIn("five", out)
        self.assertNotIn("six", out)
        self.assertNotIn("seven", out)

    def test_list_projectA_substring(self):
        """filter - list project:A /foo/"""
        code, out, err = self.t("list project:A /foo/")

        self.assertIn("one", out)
        self.assertNotIn("two", out)
        self.assertNotIn("three", out)
        self.assertNotIn("four", out)
        self.assertNotIn("five", out)
        self.assertNotIn("six", out)
        self.assertNotIn("seven", out)

    def test_list_projectA_tag(self):
        """filter - list project:A +tag"""
        code, out, err = self.t("list project:A +tag")

        self.assertIn("one", out)
        self.assertNotIn("two", out)
        self.assertNotIn("three", out)
        self.assertNotIn("four", out)
        self.assertNotIn("five", out)
        self.assertNotIn("six", out)
        self.assertNotIn("seven", out)

    def test_list_projectA_priorityH_substring(self):
        """filter - list project:A priority:H /foo/"""
        code, out, err = self.t("list project:A priority:H /foo/")

        self.assertIn("one", out)
        self.assertNotIn("two", out)
        self.assertNotIn("three", out)
        self.assertNotIn("four", out)
        self.assertNotIn("five", out)
        self.assertNotIn("six", out)
        self.assertNotIn("seven", out)

    def test_list_projectA_priorityH_tag(self):
        """filter - list project:A priority:H +tag"""
        code, out, err = self.t("list project:A priority:H +tag")

        self.assertIn("one", out)
        self.assertNotIn("two", out)
        self.assertNotIn("three", out)
        self.assertNotIn("four", out)
        self.assertNotIn("five", out)
        self.assertNotIn("six", out)
        self.assertNotIn("seven", out)

    def test_list_projectA_priorityH_substring_tag(self):
        """filter - list project:A priority:H /foo/ +tag"""
        code, out, err = self.t("list project:A priority:H /foo/ +tag")

        self.assertIn("one", out)
        self.assertNotIn("two", out)
        self.assertNotIn("three", out)
        self.assertNotIn("four", out)
        self.assertNotIn("five", out)
        self.assertNotIn("six", out)
        self.assertNotIn("seven", out)

    def test_list_projectA_priorityH_substring_tag_substring(self):
        """filter - list project:A priority:H /foo/ +tag /baz/"""
        code, out, err = self.t.runError("list project:A priority:H /foo/ +tag /baz/")

        self.assertNotIn("one", out)
        self.assertNotIn("two", out)
        self.assertNotIn("three", out)
        self.assertNotIn("four", out)
        self.assertNotIn("five", out)
        self.assertNotIn("six", out)
        self.assertNotIn("seven", out)

    def test_regex_list_project(self):
        """filter - rc.regex:on list project ~ '[A-Z]'"""
        code, out, err = self.t("rc.regex:on list project ~ \\'[A-Z]\\'")

        self.assertIn("one", out)
        self.assertIn("two", out)
        self.assertIn("three", out)
        self.assertNotIn("four", out)
        self.assertNotIn("five", out)
        self.assertNotIn("six", out)
        self.assertNotIn("seven", out)

    def test_regex_list_project_any(self):
        """filter - rc.regex:on list project~."""
        code, out, err = self.t("rc.regex:on list project ~ .")

        self.assertIn("one", out)
        self.assertIn("two", out)
        self.assertIn("three", out)
        self.assertNotIn("four", out)
        self.assertNotIn("five", out)
        self.assertNotIn("six", out)
        self.assertNotIn("seven", out)

    def test_regex_list_substring(self):
        """filter - rc.regex:on list /fo{2}/"""
        code, out, err = self.t("rc.regex:on list /fo{2}/")

        self.assertIn("one", out)
        self.assertNotIn("two", out)
        self.assertNotIn("three", out)
        self.assertNotIn("four", out)
        self.assertNotIn("five", out)
        self.assertIn("six", out)
        self.assertIn("seven", out)

    def test_regex_list_double_substring_wildcard(self):
        """filter - rc.regex:on list /f../ /b../"""
        code, out, err = self.t("rc.regex:on list /f../ /b../")

        self.assertNotIn("one", out)
        self.assertNotIn("two", out)
        self.assertNotIn("three", out)
        self.assertNotIn("four", out)
        self.assertNotIn("five", out)
        self.assertNotIn("six", out)
        self.assertIn("seven", out)

    def test_regex_list_substring_startswith(self):
        """filter - rc.regex:on list /^s/"""
        code, out, err = self.t("rc.regex:on list /^s/")

        self.assertNotIn("one", out)
        self.assertNotIn("two", out)
        self.assertNotIn("three", out)
        self.assertNotIn("four", out)
        self.assertNotIn("five", out)
        self.assertIn("six", out)
        self.assertIn("seven", out)

    def test_regex_list_substring_wildcard_startswith(self):
        """filter - rc.regex:on list /^.i/"""
        code, out, err = self.t("rc.regex:on list /^.i/")

        self.assertNotIn("one", out)
        self.assertNotIn("two", out)
        self.assertNotIn("three", out)
        self.assertNotIn("four", out)
        self.assertIn("five", out)
        self.assertIn("six", out)
        self.assertNotIn("seven", out)

    def test_regex_list_substring_or(self):
        """filter - rc.regex:on list /two|five/"""
        code, out, err = self.t("rc.regex:on list /two|five/")

        self.assertNotIn("one", out)
        self.assertIn("two", out)
        self.assertNotIn("three", out)
        self.assertNotIn("four", out)
        self.assertIn("five", out)
        self.assertNotIn("six", out)
        self.assertNotIn("seven", out)


class TestFilterDue(TestCase):
    def setUp(self):
        self.t = Task()

        self.t.config("due", "4")
        self.t.config("dateformat", "m/d/Y")

        just = datetime.datetime.now() + datetime.timedelta(days=3)
        almost = datetime.datetime.now() + datetime.timedelta(days=5)
        # NOTE: %-m and %-d are unix only
        self.just = just.strftime("%-m/%-d/%Y")
        self.almost = almost.strftime("%-m/%-d/%Y")

    def test_due_filter(self):
        """due tasks filtered correctly"""

        self.t("add one due:{0}".format(self.just))
        self.t("add two due:{0}".format(self.almost))
        self.t("add three due:today")

        code, out, err = self.t("list due:today")
        self.assertNotIn("one", out)
        self.assertNotIn("two", out)
        self.assertIn("three", out)

        code, out, err = self.t("list due.is:today")
        self.assertNotIn("one", out)
        self.assertNotIn("two", out)
        self.assertIn("three", out)


class TestBug1110(TestCase):
    def setUp(self):
        self.t = Task()

    def test_status_is_case_insensitive(self):
        """filter - status:Completed / status:completed - behave the same"""
        self.t("add ToBeCompleted")
        self.t("1 done")

        code, out, err = self.t("all status:Completed")
        self.assertIn("ToBeCompleted", out)

        code, out, err = self.t("all status:completed")
        self.assertIn("ToBeCompleted", out)


class TestBug480A(TestCase):
    @classmethod
    def setUpClass(cls):
        cls.t = Task()

        cls.t.config("defaultwidth", "0")

        cls.t("add one +ordinary")
        cls.t("add two +@strange")

    def test_long_plus_ordinary(self):
        """filter '@' in tags breaks filters: +ordinary"""
        code, out, err = self.t("long +ordinary")
        self.assertIn("one", out)
        self.assertNotIn("two", out)

    def test_long_minus_ordinary(self):
        """filter '@' in tags breaks filters: -ordinary"""
        code, out, err = self.t("long -ordinary")
        self.assertNotIn("one", out)
        self.assertIn("two", out)

    def test_long_plus_at_strange(self):
        """filter '@' in tags breaks filters: +@strange"""
        code, out, err = self.t("long +@strange")
        self.assertNotIn("one", out)
        self.assertIn("two", out)

    def test_long_minus_at_strange(self):
        """filter '@' in tags breaks filters: -@strange"""
        code, out, err = self.t("long -@strange")
        self.assertIn("one", out)
        self.assertNotIn("two", out)


class TestEmptyFilter(TestCase):
    def setUp(self):
        self.t = Task()

    def test_empty_filter_warning(self):
        """Modify tasks with no filter."""

        self.t("add foo")
        self.t("add bar")

        code, out, err = self.t.runError(
            "modify rc.allow.empty.filter=yes rc.confirmation=no priority:H"
        )
        self.assertIn("Command prevented from running.", err)

    def test_empty_filter_error(self):
        """Modify tasks with no filter, and disallowed confirmation."""

        self.t("add foo")
        self.t("add bar")

        code, out, err = self.t.runError("modify rc.allow.empty.filter=no priority:H")
        self.assertIn(
            "You did not specify a filter, and with the 'allow.empty.filter' value, no action is taken.",
            err,
        )


class TestFilterPrefix(TestCase):
    @classmethod
    def setUpClass(cls):
        """Executed once before any test in the class"""
        cls.t = Task()
        cls.t.config("verbose", "nothing")

        cls.t('add project:foo.uno  priority:H +tag "one foo"')
        cls.t('add project:foo.dos  priority:H      "two"')
        cls.t('add project:foo.tres                 "three"')
        cls.t('add project:bar.uno  priority:H      "four"')
        cls.t('add project:bar.dos             +tag "five"')
        cls.t('add project:bar.tres                 "six foo"')
        cls.t('add project:bazuno                   "seven bar foo"')
        cls.t('add project:bazdos                   "eight bar foo"')

    def test_list_all(self):
        """No filter shows all tasks."""
        code, out, err = self.t("list")
        self.assertIn("one", out)
        self.assertIn("two", out)
        self.assertIn("three", out)
        self.assertIn("four", out)
        self.assertIn("five", out)
        self.assertIn("six", out)
        self.assertIn("seven", out)
        self.assertIn("eight", out)

    def test_list_project_foo(self):
        """Filter on project name."""
        code, out, err = self.t("list project:foo")
        self.assertIn("one", out)
        self.assertIn("two", out)
        self.assertIn("three", out)
        self.assertNotIn("four", out)
        self.assertNotIn("five", out)
        self.assertNotIn("six", out)
        self.assertNotIn("seven", out)
        self.assertNotIn("eight", out)

    def test_list_project_not_foo(self):
        """Filter on not project name."""
        code, out, err = self.t("list project.not:foo")
        self.assertNotIn("one", out)
        self.assertNotIn("two", out)
        self.assertNotIn("three", out)
        self.assertIn("four", out)
        self.assertIn("five", out)
        self.assertIn("six", out)
        self.assertIn("seven", out)
        self.assertIn("eight", out)

    def test_list_project_startswith_bar(self):
        """Filter on project name start."""
        code, out, err = self.t("list project.startswith:bar")
        self.assertNotIn("one", out)
        self.assertNotIn("two", out)
        self.assertNotIn("three", out)
        self.assertIn("four", out)
        self.assertIn("five", out)
        self.assertIn("six", out)
        self.assertNotIn("seven", out)
        self.assertNotIn("eight", out)

    def test_list_project_ba(self):
        """Filter on project partial match."""
        code, out, err = self.t("list project:ba")
        self.assertNotIn("one", out)
        self.assertNotIn("two", out)
        self.assertNotIn("three", out)
        self.assertIn("four", out)
        self.assertIn("five", out)
        self.assertIn("six", out)
        self.assertIn("seven", out)
        self.assertIn("eight", out)

    def test_list_description_has_foo(self):
        """Filter on description pattern."""
        code, out, err = self.t("list description.has:foo")
        self.assertIn("one", out)
        self.assertNotIn("two", out)
        self.assertNotIn("three", out)
        self.assertNotIn("four", out)
        self.assertNotIn("five", out)
        self.assertIn("six", out)
        self.assertIn("seven", out)
        self.assertIn("eight", out)


class TestBug480B(TestCase):
    def setUp(self):
        self.t = Task()

        self.t.config("defaultwidth", "0")

        self.t("add one +t1")
        self.t("add two +t2")
        self.t("add three +t3")

    def test_numbered_tags(self):
        """filter '-t1 -t2' doesn't work"""
        code, out, err = self.t("list -t1")
        self.assertNotIn("one", out)
        self.assertIn("two", out)
        self.assertIn("three", out)

        code, out, err = self.t("list -t1 -t2")
        self.assertNotIn("one", out)
        self.assertNotIn("two", out)
        self.assertIn("three", out)

        code, out, err = self.t.runError("list -t1 -t2 -t3")
        self.assertIn("No matches", err)
        self.assertNotIn("one", out)
        self.assertNotIn("two", out)
        self.assertNotIn("three", out)

    def test_numbered_at_tags(self):
        """filter '-t1 -t2' doesn't work when '@' characters are involved"""
        self.t("1 modify +@1")
        self.t("2 modify +@2")
        self.t("3 modify +@3")

        code, out, err = self.t("list -@1")
        self.assertNotIn("one", out)
        self.assertIn("two", out)
        self.assertIn("three", out)

        code, out, err = self.t("list -@1 -@2")
        self.assertNotIn("one", out)
        self.assertNotIn("two", out)
        self.assertIn("three", out)

        code, out, err = self.t.runError("list -@1 -@2 -@3")
        self.assertIn("No matches", err)
        self.assertNotIn("one", out)
        self.assertNotIn("two", out)
        self.assertNotIn("three", out)

    def test_numbered_at_tags_punctuation(self):
        """filter '-t1 -t2' doesn't work with '@' characters and punctuation"""
        self.t("1 modify +@foo.1")
        self.t("2 modify +@foo.2")
        self.t("3 modify +@foo.3")

        code, out, err = self.t("list -@foo.1")
        self.assertNotIn("one", out)
        self.assertIn("two", out)
        self.assertIn("three", out)

        code, out, err = self.t("list -@foo.1 -@foo.2")
        self.assertNotIn("one", out)
        self.assertNotIn("two", out)
        self.assertIn("three", out)

        code, out, err = self.t.runError("list -@foo.1 -@foo.2 -@foo.3")
        self.assertIn("No matches", err)
        self.assertNotIn("one", out)
        self.assertNotIn("two", out)
        self.assertNotIn("three", out)


class TestBug485(TestCase):
    @classmethod
    def setUp(cls):
        cls.t = Task()

        cls.t.config("verbose", "nothing")

        cls.t("add one due:tomorrow recur:monthly")
        cls.t("add two due:tomorrow recur:1month")

    def test_filter_recur_monthly(self):
        """filter 'recur:monthly' doesn't list monthly tasks"""
        code, out, err = self.t("list recur:monthly")
        self.assertIn("one", out)
        self.assertIn("two", out)

    def test_filter_recur_1month(self):
        """filter 'recur:1month' doesn't list monthly tasks"""
        code, out, err = self.t("list recur:1month")
        self.assertIn("one", out)
        self.assertIn("two", out)


class TestBug489(TestCase):
    @classmethod
    def setUp(cls):
        cls.t = Task()

    def test_filter_tagless_tasks(self):
        """tags.none: filters tagless tasks"""
        self.t("add with +tag")
        self.t("add without")

        code, out, err = self.t("list tags.none:")
        self.assertNotIn("with ", out)
        self.assertIn("without", out)


class TestBug1600(TestCase):
    def setUp(self):
        self.t = Task()

    def test_filter_plus_in_descriptions(self):
        """filter - description contains +"""
        self.t("add foobar1")
        self.t("add foobar2")
        self.t("add foobar+")

        code, out, err = self.t("all")
        self.assertIn("foobar+", out)
        self.assertIn("foobar1", out)
        self.assertIn("foobar2", out)

        code, out, err = self.t(r"all description.contains:\'foobar\\+\'")
        self.assertIn("foobar+", out)
        self.assertNotIn("foobar1", out)
        self.assertNotIn("foobar2", out)

    def test_filter_question_in_descriptions(self):
        """filter - description contains ?"""
        self.t("add foobar1")
        self.t("add foo?bar")

        code, out, err = self.t("all")
        self.assertIn("foobar1", out)
        self.assertIn("foo?bar", out)

        code, out, err = self.t("all description.contains:'foo\\?bar'")
        self.assertIn("foo?bar", out)
        self.assertNotIn("foobar1", out)

    def test_filter_brackets_in_descriptions(self):
        """filter - description contains []"""
        self.t("add [foobar1]")
        self.t("add [foobar2]")

        code, out, err = self.t("all")
        self.assertIn("[foobar1]", out)
        self.assertIn("[foobar2]", out)

        code, out, err = self.t("all description.contains:'\\[foobar1\\]'")
        self.assertIn("[foobar1]", out)
        self.assertNotIn("[foobar2]", out)


class TestBug1656(TestCase):
    def setUp(self):
        self.t = Task()

    def test_report_filter_parenthesized(self):
        """default report filter parenthesized"""
        self.t("add task1 +work")
        self.t("add task2 +work")
        self.t("1 done")

        # Sanity check, next does not display completed tasks
        code, out, err = self.t("next")
        self.assertNotIn("task1", out)
        self.assertIn("task2", out)

        # The or in the filter should not cause ignoring the implicit
        # report filter
        code, out, err = self.t("next +home or +work")
        self.assertNotIn("task1", out)
        self.assertIn("task2", out)


class TestRange(TestCase):
    def setUp(self):
        """Executed before each test in the class"""
        self.t = Task()

    def test_date_range(self):
        """Verify tasks can be selected by dates ranges"""
        self.t("add one   due:2009-08-01")
        self.t("add two   due:2009-08-03")
        self.t("add three due:2009-08-05")

        code, out, err = self.t("due.after:2009-08-02 due.before:2009-08-05 ls")
        self.assertNotIn("one", out)
        self.assertIn("two", out)
        self.assertNotIn("three", out)


class TestHasHasnt(TestCase):
    def setUp(self):
        """Executed before each test in the class"""
        self.t = Task()

    def test_has_hasnt(self):
        """Verify the 'has' and 'hasnt' attribute modifiers"""
        self.t("add foo")  # 1
        self.t("add foo")  # 2
        self.t("2 annotate bar")
        self.t("add foo")  # 3
        self.t("3 annotate bar")
        self.t("3 annotate baz")
        self.t("add bar")  # 4
        self.t("add bar")  # 5
        self.t("5 annotate foo")
        self.t("add bar")  # 6
        self.t("6 annotate foo")
        self.t("6 annotate baz")
        self.t("add one")  # 7
        self.t("7 annotate two")
        self.t("7 annotate three")

        code, out, err = self.t("description.has:foo long")
        self.assertIn("\n 1", out)
        self.assertIn("\n 2", out)
        self.assertIn("\n 3", out)
        self.assertNotIn("\n 4", out)
        self.assertIn("\n 5", out)
        self.assertIn("\n 6", out)
        self.assertNotIn("\n 7", out)

        code, out, err = self.t("description.hasnt:foo long")
        self.assertNotIn("\n 1", out)
        self.assertNotIn("\n 2", out)
        self.assertNotIn("\n 3", out)
        self.assertIn("\n 4", out)
        self.assertNotIn("\n 5", out)
        self.assertNotIn("\n 6", out)
        self.assertIn("\n 7", out)


class TestBefore(TestCase):
    @classmethod
    def setUpClass(cls):
        """Executed once before any test in the class"""
        cls.t = Task()
        cls.t("add foo entry:2008-12-22 start:2008-12-22")
        cls.t("add bar entry:2009-04-17 start:2009-04-17")

    def test_correctly_recorded_start(self):
        """Verify start dates properly recorded"""
        code, out, err = self.t("_get 1.start")
        self.assertEqual(out, "2008-12-22T00:00:00\n")

        code, out, err = self.t("_get 2.start")
        self.assertEqual(out, "2009-04-17T00:00:00\n")

    def test_before_none(self):
        """Verify start.before:2008-12-01 yields nothing"""
        code, out, err = self.t("start.before:2008-12-01 _ids")
        self.assertNotIn("1", out)
        self.assertNotIn("2", out)

    def test_after_none(self):
        """Verify start.after:2009-05-01 yields nothing"""
        code, out, err = self.t("start.after:2009-05-01 _ids")
        self.assertNotIn("1", out)
        self.assertNotIn("2", out)

    def test_before_a(self):
        """Verify start.before:2009-01-01 yields '1'"""
        code, out, err = self.t("start.before:2009-01-01 _ids")
        self.assertIn("1", out)
        self.assertNotIn("2", out)

    def test_before_b(self):
        """Verify start.before:2009-05-01 yields '1' and '2'"""
        code, out, err = self.t("start.before:2009-05-01 _ids")
        self.assertIn("1", out)
        self.assertIn("2", out)

    def test_after_a(self):
        """Verify start.after:2008-12-01 yields '1' and '2'"""
        code, out, err = self.t("start.after:2008-12-01 _ids")
        self.assertIn("1", out)
        self.assertIn("2", out)

    def test_after_b(self):
        """Verify start.after:2009-01-01 yields '2'"""
        code, out, err = self.t("start.after:2009-01-01 _ids")
        self.assertNotIn("1", out)
        self.assertIn("2", out)


class TestBy(TestCase):
    def setUp(self):
        self.t = Task()

    def test_by_eoy_includes_eoy(self):
        """Verify by-end-of-year includes task due *at* end-of-year"""
        self.t("add zero due:eoy")

        code, out, err = self.t("due.by:eoy")
        self.assertIn("zero", out)

    def test_by_tomorrow_includes_tomorrow(self):
        """Verify that by-tomorrow also includes tomorrow itself"""
        self.t.faketime("2021-07-16 21:00:00")
        self.t("add zero due:2021-07-17")

        code, out, err = self.t("due.by:tomorrow")
        self.assertIn("zero", out)

    def test_by_yesterday_does_not_include_today(self):
        """Verify that by-yesterday does not include today"""
        self.t("add zero")

        code, out, err = self.t.runError("entry.by:yesterday")
        self.assertIn("No matches", err)
        self.assertNotIn("zero", out)


class Test1424(TestCase):
    def setUp(self):
        self.t = Task()

    def test_1824_days(self):
        """1424: Check that due:1824d works"""
        self.t("add foo due:1824d")
        code, out, err = self.t("_get 1.due.year")
        # NOTE This test has a possible race condition when run "during" EOY.
        # If Taskwarrior is executed at 23:59:59 on new year's eve and the
        # python code below runs at 00:00:00 on new year's day, the two will
        # disagree on the proper year. Using libfaketime with a frozen time
        # or the date set to $year-01-01 might be a good idea here.
        plus_1824d = datetime.datetime.today() + datetime.timedelta(days=1824)
        self.assertEqual(out, "%d\n" % (plus_1824d.year))

    def test_3648_days(self):
        """1424: Check that due:3648d works"""
        self.t("add foo due:3648d")
        code, out, err = self.t("_get 1.due.year")
        # NOTE This test has a possible race condition when run "during" EOY.
        # If Taskwarrior is executed at 23:59:59 on new year's eve and the
        # python code below runs at 00:00:00 on new year's day, the two will
        # disagree on the proper year. Using libfaketime with a frozen time
        # or the date set to $year-01-01 might be a good idea here.
        plus_3648d = datetime.datetime.today() + datetime.timedelta(days=3648)
        self.assertEqual(out, "%d\n" % (plus_3648d.year))


# TODO This does not look right, it adds one task, exports it, and checks the UUID.
#      The 'uuid:' filter could be ignored, and this test might pass.
class Test1452(TestCase):
    def setUp(self):
        self.t = Task()
        self.t("add task")
        self.task_uuid = self.t.export_one()["uuid"]

    def test_get_task_by_uuid_with_prefix(self):
        """1452: Tries to filter task simply by its uuid, using uuid: prefix."""
        output = self.t.export_one("uuid:%s" % self.task_uuid)

        # Sanity check it is the correct one
        self.assertEqual(output["uuid"], self.task_uuid)

    def test_get_task_by_uuid_without_prefix(self):
        """1452: Tries to filter task simply by its uuid, without using uuid: prefix."""
        output = self.t.export_one(self.task_uuid)

        # Sanity check it is the correct one
        self.assertEqual(output["uuid"], self.task_uuid)


class TestBug1456(TestCase):
    def setUp(self):
        """Executed before each test in the class"""
        self.t = Task()

    def test_quoted_expressions(self):
        """1456: Verify that a multi-term quoted filter expression works"""
        self.t("add zero")
        self.t("add one")
        self.t("add two")

        code, out, err = self.t("'/one/ or /two/' list")
        self.assertNotIn("zero", out)
        self.assertIn("one", out)
        self.assertIn("two", out)


class Test1468(TestCase):
    def setUp(self):
        self.t = Task()
        self.t("add project:home buy milk")
        self.t("add project:home mow the lawn")

    def test_single_attribute_filter(self):
        """1468: Single attribute filter (project:home)"""
        code, out, err = self.t("list project:home")
        self.assertEqual(0, code, "Exit code was non-zero ({0})".format(code))
        self.assertIn("buy milk", out)
        self.assertIn("mow the lawn", out)

    def test_attribute_and_search_filter(self):
        """1468: Attribute and implicit search filter (project:home /lawn/)"""
        code, out, err = self.t("list project:home /lawn/")
        self.assertEqual(0, code, "Exit code was non-zero ({0})".format(code))
        self.assertNotIn("buy milk", out)
        self.assertIn("mow the lawn", out)


class TestBug1521(TestCase):
    @classmethod
    def setUpClass(cls):
        """Executed once before any test in the class"""
        cls.t = Task()
        cls.t.config("verbose", "nothing")
        cls.t("add one project:WORK")
        cls.t("add two project:HOME")

    def setUp(self):
        """Executed before each test in the class"""

    def test_project_inequality(self):
        """1521: Verify that 'project.not' works"""
        code, out, err = self.t("project.not:WORK list")
        self.assertNotIn("one", out)
        self.assertIn("two", out)

    def test_project_not_equal(self):
        """1521: Verify that 'project !=' works"""
        code, out, err = self.t("project != WORK list")
        self.assertNotIn("one", out)
        self.assertIn("two", out)


class TestBug1609(TestCase):
    def setUp(self):
        """Executed before each test in the class"""
        self.t = Task()

    def test_urgency_comparison(self):
        """1609: Test that urgency is filterable"""
        self.t("add one project:P due:yesterday +tag1 +tag2")
        self.t("add two")

        code, out, err = self.t("'urgency<10' next")
        self.assertNotIn("one", out)
        self.assertIn("two", out)


class TestBug1630(TestCase):
    def setUp(self):
        """Executed before each test in the class"""
        self.t = Task()
        self.t("add zero")
        self.t("add one due:7d")
        self.t("add two due:10d")

    def test_attribute_modifier_with_duration(self):
        """1630: Verify that 'due.before:9d' is correctly interpreted"""
        code, out, err = self.t("due.before:9d list rc.verbose:nothing")
        self.assertNotIn("zero", out)
        self.assertIn("one", out)
        self.assertNotIn("two", out)

    def test_attribute_no_modifier_with_duration(self):
        """1630: Verify that 'due:7d' is correctly interpreted"""
        code, out, err = self.t("due:7d list rc.verbose:nothing")
        self.assertNotIn("zero", out)
        self.assertIn("one", out)
        self.assertNotIn("two", out)


class Test1634(TestCase):
    def setUp(self):
        self.t = Task()

        # Setup some tasks due on 2015-07-07
        self.t("add due:2015-07-07T00:00:00 ON1")
        self.t("add due:2015-07-07T14:34:56 ON2")
        self.t("add due:2015-07-07T23:59:59 ON3")

        # Setup some tasks not due on 2015-07-07
        self.t("add due:2015-07-06T23:59:59 OFF4")
        self.t("add due:2015-07-08T00:00:00 OFF5")
        self.t("add due:2015-07-08T00:00:01 OFF6")
        self.t("add due:2015-07-06T00:00:00 OFF7")

    def test_due_match_not_exact(self):
        """1634: Test that due:<date> matches any task that date."""
        code, out, err = self.t("due:2015-07-07 minimal")

        # Asswer that only tasks ON the date are listed.
        self.assertIn("ON1", out)
        self.assertIn("ON2", out)
        self.assertIn("ON3", out)

        # Assert that tasks on other dates are not listed.
        self.assertNotIn("OFF4", out)
        self.assertNotIn("OFF5", out)
        self.assertNotIn("OFF6", out)
        self.assertNotIn("OFF7", out)

    def test_due_not_match_not_exact(self):
        """1634: Test that due.not:<date> does not match any task that date."""
        code, out, err = self.t("due.not:2015-07-07 minimal")

        # Assert that task ON the date are not listed.
        self.assertNotIn("ON1", out)
        self.assertNotIn("ON2", out)
        self.assertNotIn("ON3", out)

        # Assert that tasks on other dates are listed.
        self.assertIn("OFF4", out)
        self.assertIn("OFF5", out)
        self.assertIn("OFF6", out)
        self.assertIn("OFF7", out)


class TestBug1915(TestCase):
    def setUp(self):
        """Executed before each test in the class"""
        self.t = Task()
        self.t("add project:A thingA")
        self.t("add project:B thingB")
        self.t("add project:C thingC")

    def test_complex_and_or_query_variant_one(self):
        """1915: Make sure parser handles complex and-or queries correctly (1)"""
        code, out, err = self.t(
            "rc.verbose:nothing '(project:A or project:B) and status:pending' all"
        )
        self.assertIn("thingA", out)
        self.assertIn("thingB", out)
        self.assertNotIn("thingC", out)

    def test_complex_and_or_query_variant_two(self):
        """1915: Make sure parser handles complex and-or queries correctly (2)"""
        code, out, err = self.t(
            "rc.verbose:nothing '( project:A or project:B ) and status:pending' all"
        )
        self.assertIn("thingA", out)
        self.assertIn("thingB", out)
        self.assertNotIn("thingC", out)

    @unittest.expectedFailure
    def test_complex_and_or_query_variant_three(self):
        """1915: Make sure parser handles complex and-or queries correctly (3)"""
        code, out, err = self.t(
            "rc.verbose:nothing 'status:pending and (project:A or project:B)' all"
        )
        self.assertIn("thingA", out)
        self.assertIn("thingB", out)
        self.assertNotIn("thingC", out)

    @unittest.expectedFailure
    def test_complex_and_or_query_variant_four(self):
        """1915: Make sure parser handles complex and-or queries correctly (4)"""
        code, out, err = self.t(
            "rc.verbose:nothing 'status:pending and ( project:A or project:B )' all"
        )
        self.assertIn("thingA", out)
        self.assertIn("thingB", out)
        self.assertNotIn("thingC", out)

    def test_complex_and_or_query_variant_five(self):
        """1915: Make sure parser handles complex and-or queries correctly (5)"""
        code, out, err = self.t(
            "rc.verbose:nothing status:pending and '(project:A or project:B)' all"
        )
        self.assertIn("thingA", out)
        self.assertIn("thingB", out)
        self.assertNotIn("thingC", out)

    def test_complex_and_or_query_variant_six(self):
        """1915: Make sure parser handles complex and-or queries correctly (6)"""
        code, out, err = self.t(
            "rc.verbose:nothing status:pending and '( project:A or project:B )' all"
        )
        self.assertIn("thingA", out)
        self.assertIn("thingB", out)
        self.assertNotIn("thingC", out)

    def test_complex_and_or_query_variant_seven(self):
        """1915: Make sure parser handles complex and-or queries correctly (7)"""
        code, out, err = self.t(
            "rc.verbose:nothing status:pending and \\( project:A or project:B \\) all"
        )
        self.assertIn("thingA", out)
        self.assertIn("thingB", out)
        self.assertNotIn("thingC", out)

    @unittest.expectedFailure
    def test_complex_and_or_query_variant_eight(self):
        """1915: Make sure parser handles complex and-or queries correctly (8)"""
        code, out, err = self.t(
            "rc.verbose:nothing status:pending and \\(project:A or project:B\\) all"
        )
        self.assertIn("thingA", out)
        self.assertIn("thingB", out)
        self.assertNotIn("thingC", out)


class Test2577(TestCase):
    def setUp(self):
        self.t = Task()

    def test_filtering_for_datetime_like(self):
        """2577: Check that filtering for datetime-like project names works"""
        self.t("add one pro:sat")  # looks like "saturday"
        self.t("add two pro:whatever")

        # This should not fail (fails on 2.5.3)
        code, out, err = self.t("pro:sat")

        # Assert expected output, but the crucial part of this test is success
        # of the call above
        self.assertIn("one", out)


if __name__ == "__main__":
    from simpletap import TAPTestRunner

    unittest.main(testRunner=TAPTestRunner())

# vim: ai sts=4 et sw=4 ft=python
