From f024840834fb695220ee216a5cc7eb03ba982b33 Mon Sep 17 00:00:00 2001
From: Christian Heller <c.heller@plomlompom.de>
Date: Mon, 5 Aug 2024 07:33:05 +0200
Subject: [PATCH] Simplify POST /process form field structure.

---
 plomtask/http.py       | 63 +++++++++++++-------------------
 templates/process.html | 16 ++++-----
 tests/processes.py     | 81 +++++++++++++++++-------------------------
 tests/todos.py         |  3 +-
 tests/utils.py         |  2 +-
 5 files changed, 65 insertions(+), 100 deletions(-)

diff --git a/plomtask/http.py b/plomtask/http.py
index 475f87f..fa04579 100644
--- a/plomtask/http.py
+++ b/plomtask/http.py
@@ -711,51 +711,43 @@ class TaskHandler(BaseHTTPRequestHandler):
     def do_POST_process(self, process: Process) -> str:
         """Update or insert Process of ?id= and fields defined in postvars."""
         # pylint: disable=too-many-locals
-        # pylint: disable=too-many-statements
         try:
-            title = self._form_data.get_str('title')
-            description = self._form_data.get_str('description')
-            effort = self._form_data.get_float('effort')
+            versioned = {'title': self._form_data.get_str('title'),
+                         'description': self._form_data.get_str('description'),
+                         'effort': self._form_data.get_float('effort')}
         except NotFoundException as e:
             raise BadFormatException from e
-        conditions = self._form_data.get_all_int('conditions')
-        blockers = self._form_data.get_all_int('blockers')
-        enables = self._form_data.get_all_int('enables')
-        disables = self._form_data.get_all_int('disables')
+        cond_rels = [self._form_data.get_all_int(s) for s
+                     in ['conditions', 'blockers', 'enables', 'disables']]
         calendarize = self._form_data.get_bool('calendarize')
-        suppresses = self._form_data.get_all_int('suppresses')
         step_of = self._form_data.get_all_str('step_of')
-        keep_steps = self._form_data.get_all_int('keep_step')
-        step_ids = self._form_data.get_all_int('steps')
+        suppresses = self._form_data.get_all_int('suppresses')
+        step_ids = self._form_data.get_all_int('kept_steps')
         new_top_steps = self._form_data.get_all_str('new_top_step')
-        step_process_id_to = {}
-        step_parent_id_to = {}
         new_steps_to = {}
         for step_id in step_ids:
             name = f'new_step_to_{step_id}'
             new_steps_to[step_id] = self._form_data.get_all_int(name)
-        for step_id in keep_steps:
-            name = f'step_{step_id}_process_id'
-            step_process_id_to[step_id] = self._form_data.get_int(name)
-            name = f'step_{step_id}_parent_id'
-            step_parent_id_to[step_id] = self._form_data.get_int_or_none(name)
-        process.title.set(title)
-        process.description.set(description)
-        process.effort.set(effort)
-        process.set_condition_relations(
-                self.conn, conditions, blockers, enables, disables)
+        for k, v in versioned.items():
+            getattr(process, k).set(v)
+        process.set_condition_relations(self.conn, *cond_rels)
         process.calendarize = calendarize
         process.save(self.conn)
         assert isinstance(process.id_, int)
+        # set relations to, and if non-existant yet: create, other Processes
+        # 1. owners (upwards)
+        owners_to_set = []
+        new_owner_title = None
+        for owner_identifier in step_of:
+            try:
+                owners_to_set += [int(owner_identifier)]
+            except ValueError:
+                new_owner_title = owner_identifier
+        process.set_owners(self.conn, owners_to_set)
+        # 2. owneds (downwards)
         new_step_title = None
-        steps: list[ProcessStep] = []
-        for step_id in keep_steps:
-            if step_id not in step_ids:
-                raise BadFormatException('trying to keep unknown step')
-            step = ProcessStep(step_id, process.id_,
-                               step_process_id_to[step_id],
-                               step_parent_id_to[step_id])
-            steps += [step]
+        steps: list[ProcessStep] = [ProcessStep.by_id(self.conn, step_id)
+                                    for step_id in step_ids]
         for step_id in step_ids:
             new = [ProcessStep(None, process.id_, step_process_id, step_id)
                    for step_process_id in new_steps_to[step_id]]
@@ -769,14 +761,7 @@ class TaskHandler(BaseHTTPRequestHandler):
                 new_step_title = step_identifier
         process.set_steps(self.conn, steps)
         process.set_step_suppressions(self.conn, suppresses)
-        owners_to_set = []
-        new_owner_title = None
-        for owner_identifier in step_of:
-            try:
-                owners_to_set += [int(owner_identifier)]
-            except ValueError:
-                new_owner_title = owner_identifier
-        process.set_owners(self.conn, owners_to_set)
+        # encode titles for potentially newly created Processes up or down
         params = f'id={process.id_}'
         if new_step_title:
             title_b64_encoded = b64encode(new_step_title.encode()).decode()
diff --git a/templates/process.html b/templates/process.html
index 7bb503e..a4029dc 100644
--- a/templates/process.html
+++ b/templates/process.html
@@ -17,14 +17,12 @@ details[open] > summary::after {
 
 
 
-{% macro step_with_steps(step_id, step_node, indent) %}
+{% macro step_with_steps(step_node, indent) %}
 <tr>
 <td>
-<input type="hidden" name="steps" value="{{step_id}}" />
+<input type="hidden" name="steps" value="{{step_node.step.id_}}" />
 {% if step_node.is_explicit %}
-<input type="checkbox" name="keep_step" value="{{step_id}}" checked />
-<input type="hidden" name="step_{{step_id}}_process_id" value="{{step_node.process.id_}}" />
-<input type="hidden" name="step_{{step_id}}_parent_id" value="{{step_node.parent_id or ''}}" />
+<input type="checkbox" name="kept_steps" value="{{step_node.step.id_}}" checked />
 {% endif %}
 </td>
 
@@ -60,8 +58,8 @@ details[open] > summary::after {
 {% endif %}
 </tr>
 {% if step_node.is_explicit or not step_node.seen %}
-{% for substep_id, substep in step_node.steps.items() %}
-{{ step_with_steps(substep_id, substep, indent+1) }}
+{% for substep in step_node.steps %}
+{{ step_with_steps(substep, indent+1) }}
 {% endfor %}
 {% endif %}
 {% endmacro %}
@@ -116,8 +114,8 @@ edit process of ID {{process.id_}}
 <td>
 {% if steps %}
 <table>
-{% for step_id, step_node in steps.items() %}
-{{ step_with_steps(step_id, step_node, 0) }}
+{% for step_node in steps %}
+{{ step_with_steps(step_node, 0) }}
 {% endfor %}
 </table>
 {% endif %}
diff --git a/tests/processes.py b/tests/processes.py
index 45497a6..9affa84 100644
--- a/tests/processes.py
+++ b/tests/processes.py
@@ -346,72 +346,56 @@ class TestsWithServer(TestCaseWithServer):
     def test_POST_process_steps(self) -> None:
         """Test behavior of ProcessStep posting."""
         # pylint: disable=too-many-statements
+        url = '/process?id=1'
         exp = ExpectedGetProcess(1)
         self.post_exp_process([exp], {}, 1)
-        # post first (top-level) step of proc 2 to proc 1 by 'new_top_step'
-        self.post_exp_process([exp], {}, 2)
-        self.post_exp_process([exp], {'new_top_step': 2}, 1)
+        # post first (top-level) step of proc 2 to proc 1 by 'step_of' in 2
+        self.post_exp_process([exp], {'step_of': 1}, 2)
         exp.lib_set('ProcessStep', [exp.procstep_as_dict(1, 1, 2)])
         exp.set('steps', [exp.stepnode_as_dict(1)])
-        self.check_json_get('/process?id=1', exp)
+        self.check_json_get(url, exp)
         # post empty/absent steps list to process, expect clean slate, and old
         # step to completely disappear
         self.post_exp_process([exp], {}, 1)
         exp.lib_wipe('ProcessStep')
         exp.set('steps', [])
-        self.check_json_get('/process?id=1', exp)
-        # post new step of proc2 to proc1 by 'keep_step', 'steps', and its deps
-        post = {'step_1_process_id': 2, 'step_1_parent_id': '', 'steps': [1]}
-        self.post_exp_process([exp], post | {'keep_step': [1]}, 1)
+        self.check_json_get(url, exp)
+        # post new step of proc2 to proc1 by 'new_top_step'
+        self.post_exp_process([exp], {'new_top_step': 2}, 1)
         exp.lib_set('ProcessStep', [exp.procstep_as_dict(1, 1, 2)])
+        self.post_exp_process([exp], {'kept_steps': [1]}, 1)
         exp.set('steps', [exp.stepnode_as_dict(1)])
-        self.check_json_get('/process?id=1', exp)
-        # fail with invalid or missing 'steps' dependencies
-        p_min = {'title': '', 'description': '', 'effort': 0}
-        p = p_min | {'step_1_process_id': 2, 'steps': [1], 'keep_step': [1]}
-        self.check_post(p | {'step_1_parent_id': 'foo'}, '/process?id=1', 400)
-        p = p_min | {'step_1_parent_id': '', 'steps': [1], 'keep_step': [1]}
-        self.check_post(p | {'step_1_process_id': 'foo'}, '/process?id=1', 400)
-        self.check_post(p, '/process?id=1', 400)
-        # treat valid steps data without 'keep_step' like posting emptiness
-        p = {'step_1_process_id': 2, 'step_1_parent_id': '', 'steps': [1]}
-        self.post_exp_process([exp], post, 1)
-        exp.lib_wipe('ProcessStep')
-        exp.set('steps', [])
-        self.check_json_get('/process?id=1', exp)
-        # fail when refering in 'keep_step' to step not in 'steps'
-        p = p_min | p
-        self.check_post(p | {'keep_step': [2]}, '/process?id=1', 400)
-        # expect failure independently of what parent/proc_id keys exist
-        p['step_2_process_id'], p['step_2_parent_id'] = 3, None
-        self.check_post(p | {'keep_step': [2]}, '/process?id=1', 400)
+        self.check_json_get(url, exp)
         # fail on single- and multi-step recursion
-        self.check_post(p | {'new_top_step': 1}, '/process?id=1', 400)
+        p_min = {'title': '', 'description': '', 'effort': 0}
+        self.check_post(p_min | {'new_top_step': 1}, url, 400)
+        self.check_post(p_min | {'step_of': 1}, url, 400)
         self.post_exp_process([exp], {'new_top_step': 1}, 2)
-        self.check_post(p | {'new_top_step': 2}, '/process?id=1', 400)
-        # post sibling steps
+        self.check_post(p_min | {'step_of': 2, 'new_top_step': 2}, url, 400)
         self.post_exp_process([exp], {}, 3)
-        p = {'step_1_process_id': 3, 'steps': [1], 'keep_step': [1]}
-        self.post_exp_process([exp], p | {'new_top_step': 3}, 1)
-        exp.lib_set('ProcessStep', [exp.procstep_as_dict(1, 1, 3),
-                                    exp.procstep_as_dict(2, 1, 3)])
+        self.post_exp_process([exp], {'step_of': 3}, 4)
+        self.check_post(p_min | {'new_top_step': 3, 'step_of': 4}, url, 400)
+        # post sibling steps
+        self.post_exp_process([exp], {}, 4)
+        self.post_exp_process([exp], {'new_top_step': 4}, 1)
+        self.post_exp_process([exp], {'kept_steps': [1], 'new_top_step': 4}, 1)
+        exp.lib_set('ProcessStep', [exp.procstep_as_dict(1, 1, 4),
+                                    exp.procstep_as_dict(2, 1, 4)])
         exp.set('steps', [exp.stepnode_as_dict(1), exp.stepnode_as_dict(2)])
-        self.check_json_get('/process?id=1', exp)
+        self.check_json_get(url, exp)
         # post sub-step chain
-        p = {'step_1_process_id': 3, 'step_2_process_id': 3,
-             'steps': [1, 2], 'keep_step': [1, 2]}
-        self.post_exp_process([exp], p | {'new_step_to_2': 3}, 1)
-        exp.lib_set('ProcessStep', [exp.procstep_as_dict(3, 1, 3, 2)])
+        p = {'kept_steps': [1, 2], 'new_step_to_2': 4}
+        self.post_exp_process([exp], p, 1)
+        exp.lib_set('ProcessStep', [exp.procstep_as_dict(3, 1, 4, 2)])
         exp.set('steps', [exp.stepnode_as_dict(1),
                           exp.stepnode_as_dict(2, steps=[
                               exp.stepnode_as_dict(3)])])
-        self.check_json_get('/process?id=1', exp)
-        p['steps'], p['keep_step'] = [1, 2, 3], [1, 2, 3]
-        p['step_3_process_id'] = 3
-        # fail post sub-step that would cause recursion
-        self.post_exp_process([exp], {'new_top_step': 1}, 2)
-        exp.lib_set('ProcessStep', [exp.procstep_as_dict(4, 2, 1)])
-        self.check_post(p_min | p | {'new_step_to_2': 2}, '/process?id=1', 400)
+        self.check_json_get(url, exp)
+        # fail posting sub-step that would cause recursion
+        self.post_exp_process([exp], {}, 6)
+        self.post_exp_process([exp], {'new_top_step': 6}, 5)
+        p = p_min | {'kept_steps': [1, 2, 3], 'new_step_to_2': 5, 'step_of': 6}
+        self.check_post(p, url, 400)
 
     def test_GET(self) -> None:
         """Test /process and /processes response codes."""
@@ -455,8 +439,7 @@ class TestsWithServer(TestCaseWithServer):
         self.post_exp_process([exp], proc2_post | {'new_top_step': [1]}, 2)
         proc3_post = {'title': 'baz', 'description': 'zab', 'effort': 0.9}
         self.post_exp_process([exp], proc3_post | {'new_top_step': [1]}, 3)
-        proc3_post = proc3_post | {'new_top_step': [2], 'keep_step': [2],
-                                   'steps': [2], 'step_2_process_id': 1}
+        proc3_post = proc3_post | {'new_top_step': [2], 'kept_steps': [2]}
         self.post_exp_process([exp], proc3_post, 3)
         exp.lib_set('ProcessStep', [exp.procstep_as_dict(1, 2, 1),
                                     exp.procstep_as_dict(2, 3, 1),
diff --git a/tests/todos.py b/tests/todos.py
index 41adca9..752e289 100644
--- a/tests/todos.py
+++ b/tests/todos.py
@@ -523,8 +523,7 @@ class TestsWithServer(TestCaseWithServer):
         exp.set('steps_todo_to_process', [step1_proc2])
         self.check_json_get('/todo?id=1', exp)
         # test display of parallel chains
-        proc_steps_post = {'new_top_step': 4, 'keep_step': [1],
-                           'step_1_process_id': 2, 'steps': [1, 4]}
+        proc_steps_post = {'new_top_step': 4, 'kept_steps': [1, 3]}
         self.post_exp_process([], proc_steps_post, 1)
         step4_proc4 = exp.step_as_dict(4, [], 4, fillable=True)
         exp.lib_set('ProcessStep', [exp.procstep_as_dict(4, 1, 4, None)])
diff --git a/tests/utils.py b/tests/utils.py
index 3d86163..df3feea 100644
--- a/tests/utils.py
+++ b/tests/utils.py
@@ -826,7 +826,7 @@ class Expected:
             proc = self.proc_as_dict(id_,
                                      d['title'], d['description'], d['effort'])
         ignore = {'title', 'description', 'effort', 'new_top_step', 'step_of',
-                  'keep_step', 'steps'}
+                  'kept_steps'}
         for k, v in d.items():
             if k in ignore\
                     or k.startswith('step_') or k.startswith('new_step_to'):
-- 
2.30.2