mirror of
https://github.com/GothenburgBitFactory/taskwarrior.git
synced 2025-06-26 10:54:26 +02:00

* Issue warnings instead of errors for 'weird' tasks * Support more comprehensive checks when adding a task
357 lines
13 KiB
Python
Executable file
357 lines
13 KiB
Python
Executable file
#!/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 json
|
|
|
|
# Ensure python finds the local simpletap module
|
|
sys.path.append(os.path.dirname(os.path.abspath(__file__)))
|
|
|
|
from basetest import Task, TestCase
|
|
from basetest.utils import mkstemp
|
|
|
|
|
|
class TestImport(TestCase):
|
|
def setUp(self):
|
|
self.t = Task()
|
|
self.t.config("dateformat", "m/d/Y")
|
|
|
|
# Multiple tasks.
|
|
self.data1 = """[
|
|
{"uuid":"a0000000-a000-a000-a000-a00000000000","description":"zero","project":"A","status":"pending","entry":"1234567889"},
|
|
{"uuid":"a1111111-a111-a111-a111-a11111111111","description":"one","project":"B","status":"pending","entry":"1234567889"},
|
|
{"uuid":"a2222222-a222-a222-a222-a22222222222","description":"two","status":"completed","entry":"1234524689","end":"1234524690"}
|
|
]
|
|
"""
|
|
|
|
# Single task.
|
|
self.data2 = """{"uuid":"44444444-4444-4444-4444-444444444444","description":"three","status":"pending","entry":"1234567889"}
|
|
"""
|
|
|
|
# Free-form JSON.
|
|
self.data3 = """
|
|
|
|
{
|
|
"uuid"
|
|
:
|
|
"55555555-5555-5555-5555-555555555555"
|
|
,
|
|
"description"
|
|
:
|
|
"four"
|
|
,
|
|
"status"
|
|
:
|
|
"pending"
|
|
,
|
|
"entry"
|
|
:
|
|
"1234567889"
|
|
}
|
|
|
|
"""
|
|
|
|
def assertData1(self):
|
|
code, out, err = self.t("list")
|
|
self.assertRegex(out, "1.+A.+zero")
|
|
self.assertRegex(out, "2.+B.+one")
|
|
self.assertNotIn("two", out)
|
|
|
|
code, out, err = self.t("completed")
|
|
self.assertNotIn("zero", out)
|
|
self.assertNotIn("one", out)
|
|
# complete has completion date as 1st column
|
|
self.assertRegex(out, "2/13/2009.+two")
|
|
|
|
def assertData2(self):
|
|
code, out, err = self.t("list")
|
|
self.assertRegex(out, "3.+three")
|
|
|
|
def assertData3(self):
|
|
code, out, err = self.t("list")
|
|
self.assertIn("four", out)
|
|
|
|
def test_import_stdin(self):
|
|
"""Import from stdin"""
|
|
code, out, err = self.t("import -", input=self.data1)
|
|
self.assertIn("Imported 3 tasks", err)
|
|
|
|
self.assertData1()
|
|
|
|
def test_import_stdin_default(self):
|
|
"""Import from stdin is default"""
|
|
code, out, err = self.t("import", input=self.data1)
|
|
self.assertIn("Imported 3 tasks", err)
|
|
|
|
self.assertData1()
|
|
|
|
def test_import_file(self):
|
|
"""Import from a file"""
|
|
filename = mkstemp(self.data1)
|
|
|
|
code, out, err = self.t("import {0}".format(filename))
|
|
self.assertIn("Imported 3 tasks", err)
|
|
|
|
self.assertData1()
|
|
|
|
def test_double_import(self):
|
|
"""Multiple imports persist data"""
|
|
code, out, err = self.t("import -", input=self.data1)
|
|
self.assertIn("Imported 3 tasks", err)
|
|
|
|
code, out, err = self.t("import -", input=self.data2)
|
|
self.assertIn("Imported 1 tasks", err)
|
|
|
|
self.assertData1()
|
|
self.assertData2()
|
|
|
|
def test_freeform_import(self):
|
|
"""Import JSON with arbitrary formatting"""
|
|
code, out, err = self.t("import -", input=self.data3)
|
|
self.assertIn("Imported 1 tasks", err)
|
|
|
|
self.assertData3()
|
|
|
|
def test_import_update(self):
|
|
"""Update existing tasks"""
|
|
self.t("import", input=self.data1)
|
|
self.t("a1111111-a111-a111-a111-a11111111111 delete", input="y\n")
|
|
self.t("next") # Run GC
|
|
|
|
_t = sorted(self.t.export(), key=lambda t: t["uuid"])
|
|
_t[0]["project"] = "C"
|
|
_t[1]["status"] = "pending"
|
|
_t[2]["status"] = "pending"
|
|
|
|
self.t("import", input="\n".join(json.dumps(t) for t in _t))
|
|
|
|
_t = sorted(self.t.export(), key=lambda t: t["uuid"])
|
|
self.assertEqual(_t[0]["status"], "pending")
|
|
self.assertEqual(_t[0]["project"], "C")
|
|
self.assertEqual(_t[1]["status"], "pending")
|
|
self.assertEqual(_t[2]["status"], "pending")
|
|
|
|
def test_import_python_json(self):
|
|
"""Python's default JSON formatting"""
|
|
_t = json.loads(self.data1)
|
|
code, out, err = self.t("import", input=json.dumps(_t))
|
|
self.assertIn("Imported 3 tasks", err)
|
|
self.assertData1()
|
|
|
|
def test_import_no_newlines(self):
|
|
"""JSON array without newlines"""
|
|
code, out, err = self.t("import", input=self.data1.replace("\n", ""))
|
|
self.assertIn("Imported 3 tasks", err)
|
|
self.assertData1()
|
|
|
|
def test_import_newlines(self):
|
|
"""JSON array with newlines after each value"""
|
|
_t = json.loads(self.data1)
|
|
code, out, err = self.t("import", input=json.dumps(_t, indent=0))
|
|
self.assertIn("Imported 3 tasks", err)
|
|
self.assertData1()
|
|
|
|
def test_import_newlines_whitespace(self):
|
|
"""JSON array with whitespace before and after names and values"""
|
|
_data = """[
|
|
{ "uuid":"a0000000-a000-a000-a000-a00000000000" , "description" : "zero" ,"project":"A", "status":"pending","entry":"1234567889" } ,
|
|
{ "uuid":"a1111111-a111-a111-a111-a11111111111","description":"one","project":"B","status":"pending","entry":"1234567889"}, {"uuid":"a2222222-a222-a222-a222-a22222222222","description":"two","status":"completed","entry":"1234524689","end":"1234524690" }
|
|
]"""
|
|
code, out, err = self.t("import", input=_data)
|
|
self.assertIn("Imported 3 tasks", err)
|
|
self.assertData1()
|
|
|
|
def test_import_old_depends(self):
|
|
"""Several dependencies used to be a comma seperated string"""
|
|
_data = """{"uuid":"a0000000-a000-a000-a000-a00000000000","depends":"a1111111-a111-a111-a111-a11111111111,a2222222-a222-a222-a222-a22222222222","description":"zero","project":"A","status":"pending","entry":"1234567889"}"""
|
|
self.t("import", input=self.data1)
|
|
self.t("import", input=_data)
|
|
_t = self.t.export("a0000000-a000-a000-a000-a00000000000")[0]
|
|
self.assertIn("a1111111-a111-a111-a111-a11111111111", _t["depends"])
|
|
self.assertIn("a2222222-a222-a222-a222-a22222222222", _t["depends"])
|
|
|
|
def test_import_new_depend(self):
|
|
"""One dependency is a single array element"""
|
|
_data = """{"uuid":"a0000000-a000-a000-a000-a00000000000","depends":["a1111111-a111-a111-a111-a11111111111"],"description":"zero","project":"A","status":"pending","entry":"1234567889"}"""
|
|
self.t("import", input=self.data1)
|
|
self.t("import", input=_data)
|
|
_t = self.t.export("a0000000-a000-a000-a000-a00000000000")[0]
|
|
self.assertEqual(_t["depends"][0], "a1111111-a111-a111-a111-a11111111111")
|
|
|
|
def test_import_new_depends(self):
|
|
"""Several dependencies are an array"""
|
|
_data = """{"uuid":"a0000000-a000-a000-a000-a00000000000","depends":["a1111111-a111-a111-a111-a11111111111","a2222222-a222-a222-a222-a22222222222"],"description":"zero","project":"A","status":"pending","entry":"1234567889"}"""
|
|
self.t("import", input=self.data1)
|
|
self.t("import", input=_data)
|
|
_t = self.t.export("a0000000-a000-a000-a000-a00000000000")[0]
|
|
|
|
for _uuid in [
|
|
"a1111111-a111-a111-a111-a11111111111",
|
|
"a2222222-a222-a222-a222-a22222222222",
|
|
]:
|
|
self.assertTrue((_t["depends"][0] == _uuid) or (_t["depends"][1] == _uuid))
|
|
|
|
def test_import_same_task_twice(self):
|
|
"""Test import same task twice"""
|
|
_data = (
|
|
"""{"uuid":"a1111111-a222-a333-a444-a55555555555","description":"data4"}"""
|
|
)
|
|
self.t("import", input=_data)
|
|
code, out1, err = self.t("export")
|
|
self.t.faketime("+1s")
|
|
self.t("import", input=_data)
|
|
code, out2, err = self.t("export")
|
|
self.assertEqual(out1, out2)
|
|
|
|
def test_import_duplicate_uuid_in_input(self):
|
|
"""Test import warns if input contains the same UUID twice."""
|
|
_data = """[
|
|
{"uuid":"a0000000-a000-a000-a000-a00000000000","description":"first description"},
|
|
{"uuid":"a0000000-a000-a000-a000-a00000000000","description":"second description"}
|
|
]"""
|
|
_, _, err = self.t("import", input=_data)
|
|
self.assertIn(
|
|
"Input contains UUID 'a0000000-a000-a000-a000-a00000000000' 2 times", err
|
|
)
|
|
|
|
|
|
class TestImportExportRoundtrip(TestCase):
|
|
def setUp(self):
|
|
self.t1 = Task()
|
|
self.t2 = Task()
|
|
|
|
for client in (self.t1, self.t2):
|
|
client.config("dateformat", "m/d/Y")
|
|
client.config("verbose", "0")
|
|
client.config("defaultwidth", "100")
|
|
|
|
def _validate_data(self, client):
|
|
code, out, err = client("_get 1.priority")
|
|
self.assertEqual("H\n", out)
|
|
code, out, err = client("_get 1.project")
|
|
self.assertEqual("A\n", out)
|
|
code, out, err = client("_get 1.description")
|
|
self.assertEqual("one/1\n", out)
|
|
code, out, err = client("_get 2.tags")
|
|
self.assertEqual("tag1,tag2\n", out)
|
|
code, out, err = client("_get 2.description")
|
|
self.assertEqual("two\n", out)
|
|
|
|
def test_import_export(self):
|
|
"""Test importing exported data"""
|
|
self.t1("add priority:H project:A -- one/1")
|
|
self.t1("add +tag1 +tag2 two")
|
|
|
|
code, out1, err = self.t1("export")
|
|
|
|
self.t2("import -", input=out1)
|
|
code, out2, err = self.t2("export")
|
|
|
|
self.assertEqual(out1, out2)
|
|
|
|
self._validate_data(self.t1)
|
|
self._validate_data(self.t2)
|
|
|
|
|
|
class TestImportValidate(TestCase):
|
|
def setUp(self):
|
|
self.t = Task()
|
|
|
|
def test_import_empty_json(self):
|
|
"""Verify empty JSON is ignored"""
|
|
j = "{}"
|
|
code, out, err = self.t.runError("import", input=j)
|
|
self.assertIn("Cannot import an empty task.", err)
|
|
|
|
def test_import_invalid_uuid(self):
|
|
"""Verify invalid UUID is caught"""
|
|
j = '{"uuid":"1", "description":"bad"}'
|
|
code, out, err = self.t.runError("import", input=j)
|
|
self.assertIn("Not a valid UUID", err)
|
|
|
|
def test_import_invalid_uuid_array(self):
|
|
"""Verify invalid UUID is caught in array form"""
|
|
j = '[{"uuid":"1", "description":"bad"}]'
|
|
code, out, err = self.t.runError("import", input=j)
|
|
self.assertIn("Not a valid UUID", err)
|
|
|
|
def test_import_invalid_uuid2(self):
|
|
"""Verify invalid UUID is caught, part two"""
|
|
# UUID is the right length, but with s/-/0/.
|
|
j = '{"uuid":"a1a1a1a10a1a10a1a10a1a10a1a1a1a1a1a1", "description":"bad"}'
|
|
code, out, err = self.t.runError("import", input=j)
|
|
self.assertIn("Not a valid UUID", err)
|
|
|
|
def test_import_invalid_status(self):
|
|
"""Verify invalid status is caught"""
|
|
j = '{"status":"foo", "description":"bad"}'
|
|
code, out, err = self.t.runError("import", input=j)
|
|
self.assertIn("The status 'foo' is not valid.", err)
|
|
|
|
def test_import_malformed_annotation(self):
|
|
"""Verify invalid 'annnotations' is caught"""
|
|
j = '{"description": "bad", "annotations": "bad"}'
|
|
code, out, err = self.t.runError("import", input=j)
|
|
self.assertIn('Annotations is malformed: "bad"', err)
|
|
|
|
|
|
class TestImportWithoutISO(TestCase):
|
|
def setUp(self):
|
|
self.t = Task()
|
|
|
|
def test_import_with_iso_enabled(self):
|
|
j = '{"uuid":"a2a2a2a2-a2a2-a2a2-a2a2-a2a2a2a2a2a2", "description":"one", "entry":"20151018T144200"}'
|
|
self.t("import rc.date.iso=1", input=j)
|
|
code, out, err = self.t("_get 1.entry")
|
|
self.assertIn("2015-10-18T14:42:00\n", out)
|
|
|
|
def test_import_with_iso_disabled(self):
|
|
j = '{"uuid":"a2a2a2a2-a2a2-a2a2-a2a2-a2a2a2a2a2a2", "description":"one", "entry":"20151018T144200"}'
|
|
self.t("import rc.date.iso=0", input=j)
|
|
code, out, err = self.t("_get 1.entry")
|
|
self.assertIn("2015-10-18T14:42:00\n", out)
|
|
|
|
|
|
class TestBug1441(TestCase):
|
|
def setUp(self):
|
|
self.t = Task()
|
|
|
|
def test_import_filename(self):
|
|
"""1441: import fails if file doesn't exist"""
|
|
code, out, err = self.t.runError("import xxx_doesnotexist")
|
|
|
|
self.assertIn("File 'xxx_doesnotexist' not found.", err)
|
|
|
|
|
|
if __name__ == "__main__":
|
|
from simpletap import TAPTestRunner
|
|
|
|
unittest.main(testRunner=TAPTestRunner())
|
|
|
|
# vim: ai sts=4 et sw=4 ft=python
|