home · contact · privacy
Refactor from_table_row methods of core DB models.
[plomtask] / tests / todos.py
1 """Test Todos module."""
2 from tests.utils import TestCaseWithDB, TestCaseWithServer
3 from plomtask.todos import Todo
4 from plomtask.processes import Process
5 from plomtask.conditions import Condition
6 from plomtask.exceptions import (NotFoundException, BadFormatException,
7                                  HandledException)
8
9
10 class TestsWithDB(TestCaseWithDB):
11     """Tests requiring DB, but not server setup."""
12
13     def setUp(self) -> None:
14         super().setUp()
15         self.date1 = '2024-01-01'
16         self.date2 = '2024-01-02'
17         self.proc = Process(None)
18         self.proc.save(self.db_conn)
19         self.cond1 = Condition(None)
20         self.cond1.save(self.db_conn)
21         self.cond2 = Condition(None)
22         self.cond2.save(self.db_conn)
23
24     def test_Todo_by_id(self) -> None:
25         """Test creation and findability of Todos."""
26         process_unsaved = Process(None)
27         todo = Todo(None, process_unsaved, False, self.date1)
28         with self.assertRaises(NotFoundException):
29             todo.save(self.db_conn)
30         process_unsaved.save(self.db_conn)
31         todo.save(self.db_conn)
32         self.assertEqual(Todo.by_id(self.db_conn, 1), todo)
33         with self.assertRaises(NotFoundException):
34             self.assertEqual(Todo.by_id(self.db_conn, 0), todo)
35         with self.assertRaises(NotFoundException):
36             self.assertEqual(Todo.by_id(self.db_conn, 2), todo)
37
38     def test_Todo_by_date(self) -> None:
39         """Test findability of Todos by date."""
40         t1 = Todo(None, self.proc, False, self.date1)
41         t1.save(self.db_conn)
42         t2 = Todo(None, self.proc, False, self.date1)
43         t2.save(self.db_conn)
44         self.assertEqual(Todo.by_date(self.db_conn, self.date1), [t1, t2])
45         self.assertEqual(Todo.by_date(self.db_conn, self.date2), [])
46         self.assertEqual(Todo.by_date(self.db_conn, 'foo'), [])
47
48     def test_Todo_from_process(self) -> None:
49         """Test spawning of Todo attributes from Process."""
50         assert isinstance(self.cond1.id_, int)
51         assert isinstance(self.cond2.id_, int)
52         self.proc.set_conditions(self.db_conn, [self.cond1.id_])
53         todo = Todo(None, self.proc, False, self.date1)
54         self.assertEqual(todo.conditions, [self.cond1])
55         todo.set_conditions(self.db_conn, [self.cond2.id_])
56         self.assertEqual(todo.conditions, [self.cond2])
57         self.assertEqual(self.proc.conditions, [self.cond1])
58         self.proc.set_fulfills(self.db_conn, [self.cond1.id_])
59         todo = Todo(None, self.proc, False, self.date1)
60         self.assertEqual(todo.fulfills, [self.cond1])
61         todo.set_fulfills(self.db_conn, [self.cond2.id_])
62         self.assertEqual(todo.fulfills, [self.cond2])
63         self.assertEqual(self.proc.fulfills, [self.cond1])
64         self.proc.set_undoes(self.db_conn, [self.cond1.id_])
65         todo = Todo(None, self.proc, False, self.date1)
66         self.assertEqual(todo.undoes, [self.cond1])
67         todo.set_undoes(self.db_conn, [self.cond2.id_])
68         self.assertEqual(todo.undoes, [self.cond2])
69         self.assertEqual(self.proc.undoes, [self.cond1])
70
71     def test_Todo_on_conditions(self) -> None:
72         """Test effect of Todos on Conditions."""
73         assert isinstance(self.cond1.id_, int)
74         assert isinstance(self.cond2.id_, int)
75         todo = Todo(None, self.proc, False, self.date1)
76         todo.save(self.db_conn)
77         todo.set_fulfills(self.db_conn, [self.cond1.id_])
78         todo.set_undoes(self.db_conn, [self.cond2.id_])
79         todo.is_done = True
80         self.assertEqual(self.cond1.is_active, True)
81         self.assertEqual(self.cond2.is_active, False)
82         todo.is_done = False
83         self.assertEqual(self.cond1.is_active, True)
84         self.assertEqual(self.cond2.is_active, False)
85
86     def test_Todo_enablers_disablers(self) -> None:
87         """Test Todo.enablers_for_at/disablers_for_at."""
88         assert isinstance(self.cond1.id_, int)
89         assert isinstance(self.cond2.id_, int)
90         todo1 = Todo(None, self.proc, False, self.date1)
91         todo1.save(self.db_conn)
92         todo1.set_fulfills(self.db_conn, [self.cond1.id_])
93         todo1.set_undoes(self.db_conn, [self.cond2.id_])
94         todo1.save(self.db_conn)
95         todo2 = Todo(None, self.proc, False, self.date1)
96         todo2.save(self.db_conn)
97         todo2.set_fulfills(self.db_conn, [self.cond2.id_])
98         todo2.save(self.db_conn)
99         todo3 = Todo(None, self.proc, False, self.date2)
100         todo3.save(self.db_conn)
101         todo3.set_fulfills(self.db_conn, [self.cond2.id_])
102         todo3.save(self.db_conn)
103         enablers = Todo.enablers_for_at(self.db_conn, self.cond1, self.date1)
104         self.assertEqual(enablers, [todo1])
105         enablers = Todo.enablers_for_at(self.db_conn, self.cond1, self.date2)
106         self.assertEqual(enablers, [])
107         disablers = Todo.disablers_for_at(self.db_conn, self.cond1, self.date1)
108         self.assertEqual(disablers, [])
109         disablers = Todo.disablers_for_at(self.db_conn, self.cond1, self.date2)
110         self.assertEqual(disablers, [])
111         enablers = Todo.enablers_for_at(self.db_conn, self.cond2, self.date1)
112         self.assertEqual(enablers, [todo2])
113         enablers = Todo.enablers_for_at(self.db_conn, self.cond2, self.date2)
114         self.assertEqual(enablers, [todo3])
115         disablers = Todo.disablers_for_at(self.db_conn, self.cond2, self.date1)
116         self.assertEqual(disablers, [todo1])
117         disablers = Todo.disablers_for_at(self.db_conn, self.cond2, self.date2)
118         self.assertEqual(disablers, [])
119
120     def test_Todo_children(self) -> None:
121         """Test Todo.children relations."""
122         todo_1 = Todo(None, self.proc, False, self.date1)
123         todo_2 = Todo(None, self.proc, False, self.date1)
124         with self.assertRaises(HandledException):
125             todo_1.add_child(todo_2)
126         todo_1.save(self.db_conn)
127         with self.assertRaises(HandledException):
128             todo_1.add_child(todo_2)
129         todo_2.save(self.db_conn)
130         todo_1.add_child(todo_2)
131         todo_1.save(self.db_conn)
132         assert isinstance(todo_1.id_, int)
133         todo_retrieved = Todo.by_id(self.db_conn, todo_1.id_)
134         self.assertEqual(todo_retrieved.children, [todo_2])
135         with self.assertRaises(BadFormatException):
136             todo_2.add_child(todo_1)
137
138     def test_Todo_conditioning(self) -> None:
139         """Test Todo.doability conditions."""
140         assert isinstance(self.cond1.id_, int)
141         todo_1 = Todo(None, self.proc, False, self.date1)
142         todo_1.save(self.db_conn)
143         todo_2 = Todo(None, self.proc, False, self.date1)
144         todo_2.save(self.db_conn)
145         todo_2.add_child(todo_1)
146         with self.assertRaises(BadFormatException):
147             todo_2.is_done = True
148         todo_1.is_done = True
149         todo_2.is_done = True
150         todo_2.is_done = False
151         todo_2.set_conditions(self.db_conn, [self.cond1.id_])
152         with self.assertRaises(BadFormatException):
153             todo_2.is_done = True
154         self.cond1.is_active = True
155         todo_2.is_done = True
156
157     def test_Todo_singularity(self) -> None:
158         """Test pointers made for single object keep pointing to it."""
159         todo = Todo(None, self.proc, False, self.date1)
160         todo.save(self.db_conn)
161         retrieved_todo = Todo.by_id(self.db_conn, 1)
162         todo.is_done = True
163         self.assertEqual(retrieved_todo.is_done, True)
164         retrieved_todo = Todo.by_date(self.db_conn, self.date1)[0]
165         retrieved_todo.is_done = False
166         self.assertEqual(todo.is_done, False)
167
168
169 class TestsWithServer(TestCaseWithServer):
170     """Tests against our HTTP server/handler (and database)."""
171
172     def test_do_POST_day(self) -> None:
173         """Test Todo posting of POST /day."""
174         form_data = {'title': '', 'description': '', 'effort': 1}
175         self.check_post(form_data, '/process?id=', 302, '/')
176         self.check_post(form_data, '/process?id=', 302, '/')
177         proc = Process.by_id(self.db_conn, 1)
178         proc2 = Process.by_id(self.db_conn, 2)
179         form_data = {'comment': ''}
180         self.check_post(form_data, '/day?date=2024-01-01', 302, '/')
181         self.assertEqual(Todo.by_date(self.db_conn, '2024-01-01'), [])
182         form_data['new_todo'] = str(proc.id_)
183         self.check_post(form_data, '/day?date=2024-01-01', 302, '/')
184         todos = Todo.by_date(self.db_conn, '2024-01-01')
185         self.assertEqual(1, len(todos))
186         todo1 = todos[0]
187         self.assertEqual(todo1.id_, 1)
188         self.assertEqual(todo1.process.id_, proc.id_)
189         self.assertEqual(todo1.is_done, False)
190         form_data['new_todo'] = str(proc2.id_)
191         self.check_post(form_data, '/day?date=2024-01-01', 302, '/')
192         todos = Todo.by_date(self.db_conn, '2024-01-01')
193         todo1 = todos[1]
194         self.assertEqual(todo1.id_, 2)
195         self.assertEqual(todo1.process.id_, proc2.id_)
196         self.assertEqual(todo1.is_done, False)
197
198     def test_do_POST_todo(self) -> None:
199         """Test POST /todo."""
200         def post_and_reload(form_data: dict[str, object],
201                             status: int = 302) -> Todo:
202             self.check_post(form_data, '/todo?id=1', status, '/')
203             self.db_conn.cached_todos = {}
204             return Todo.by_date(self.db_conn, '2024-01-01')[0]
205         form_data = {'title': '', 'description': '', 'effort': 1}
206         self.check_post(form_data, '/process', 302, '/')
207         form_data = {'comment': '', 'new_todo': 1}
208         self.check_post(form_data, '/day?date=2024-01-01', 302, '/')
209         form_data = {}
210         self.check_post(form_data, '/todo=', 404)
211         self.check_post(form_data, '/todo?id=', 404)
212         self.check_post(form_data, '/todo?id=FOO', 400)
213         self.check_post(form_data, '/todo?id=0', 404)
214         todo1 = post_and_reload(form_data)
215         self.assertEqual(todo1.children, [])
216         self.assertEqual(todo1.parents, [])
217         self.assertEqual(todo1.is_done, False)
218         form_data = {'done': ''}
219         todo1 = post_and_reload(form_data)
220         self.assertEqual(todo1.is_done, True)
221         form_data = {}
222         todo1 = post_and_reload(form_data)
223         self.assertEqual(todo1.is_done, False)
224         form_data = {'adopt': 'foo'}
225         self.check_post(form_data, '/todo?id=1', 400)
226         form_data = {'adopt': 1}
227         self.check_post(form_data, '/todo?id=1', 400)
228         form_data = {'adopt': 2}
229         self.check_post(form_data, '/todo?id=1', 404)
230         form_data = {'comment': '', 'new_todo': 1}
231         self.check_post(form_data, '/day?date=2024-01-01', 302, '/')
232         form_data = {'adopt': 2}
233         todo1 = post_and_reload(form_data)
234         todo2 = Todo.by_date(self.db_conn, '2024-01-01')[1]
235         self.assertEqual(todo1.children, [todo2])
236         self.assertEqual(todo1.parents, [])
237         self.assertEqual(todo2.children, [])
238         self.assertEqual(todo2.parents, [todo1])
239         self.check_post(form_data, '/todo?id=1', 400, '/')
240         form_data = {'done': ''}
241         todo1 = post_and_reload(form_data, 400)
242         self.assertEqual(todo1.is_done, False)
243
244     def test_do_GET_todo(self) -> None:
245         """Test GET /todo response codes."""
246         form_data = {'title': '', 'description': '', 'effort': 1}
247         self.check_post(form_data, '/process?id=', 302, '/')
248         form_data = {'comment': '', 'new_todo': 1}
249         self.check_post(form_data, '/day?date=2024-01-01', 302, '/')
250         self.check_get('/todo', 404)
251         self.check_get('/todo?id=', 404)
252         self.check_get('/todo?id=foo', 400)
253         self.check_get('/todo?id=0', 404)
254         self.check_get('/todo?id=1', 200)