#!/usr/bin/env python3 # -*- coding: utf-8 -*- ############################################################################### # # 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: 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: 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