From 9c5395cf3ff62c5347cf952468310da4cc6620ac Mon Sep 17 00:00:00 2001 From: Sergei Maertens Date: Thu, 2 Jan 2025 16:54:17 +0100 Subject: [PATCH 1/3] :test_tube: [#4969] Add regression test for accidental counter reset --- .../tests/e2e_tests/test_form_designer.py | 47 +++++++++++++++++++ 1 file changed, 47 insertions(+) diff --git a/src/openforms/forms/tests/e2e_tests/test_form_designer.py b/src/openforms/forms/tests/e2e_tests/test_form_designer.py index 398eaa004c..685c5553c6 100644 --- a/src/openforms/forms/tests/e2e_tests/test_form_designer.py +++ b/src/openforms/forms/tests/e2e_tests/test_form_designer.py @@ -17,6 +17,7 @@ from openforms.utils.tests.cache import clear_caches from openforms.variables.constants import FormVariableDataTypes, FormVariableSources +from ...models import Form from ..factories import ( FormDefinitionFactory, FormFactory, @@ -1300,6 +1301,52 @@ def setUpTestData(): await page.keyboard.press("ArrowDown") await expect(page.get_by_text("Field 2 (field2)")).to_be_visible() + @tag("gh-4969") + async def test_saving_form_does_not_reset_submission_counter(self): + @sync_to_async + def setUpTestData(): + # set up a form + form = FormFactory.create(generate_minimal_setup=True, submission_counter=0) + return form + + @sync_to_async + def update_submission_counter(form: Form): + form.submission_counter = 10 + form.save(update_fields=["submission_counter"]) + + await create_superuser() + form = await setUpTestData() + admin_url = str( + furl(self.live_server_url) + / reverse("admin:forms_form_change", args=(form.pk,)) + ) + + async with browser_page() as page: + await self._admin_login(page) + await page.goto(str(admin_url)) + await expect( + page.get_by_role("tab", name="Steps and fields") + ).to_be_visible() + + # now, after loading the form, modify the submission counter, simulating some + # submissions happened while the form was opened in some admin screen. + await update_submission_counter(form) + + # Save form + await page.get_by_role("button", name="Save", exact=True).click() + changelist_url = str( + furl(self.live_server_url) / reverse("admin:forms_form_changelist") + ) + await expect(page).to_have_url(changelist_url) + + @sync_to_async + def assert_state(form: Form): + form.refresh_from_db() + + self.assertEqual(form.submission_counter, 10) + + await assert_state(form) + class FormDesignerTooltipTests(E2ETestCase): async def test_tooltip_fields_are_present(self): From 0440daa2f8271dfdc6bbeed1d9a92750816dd502 Mon Sep 17 00:00:00 2001 From: Sergei Maertens Date: Thu, 2 Jan 2025 17:11:42 +0100 Subject: [PATCH 2/3] :bug: [#4969] Exclude the submission counter when saving a form If the GET data is stale while the form is loaded in the admin, counter updates done between loading-and-saving are dropped and overwritten with the stale counter. Dropping this key entirely (since it's optional anyway) is the easiest way to prevent this. --- .../js/components/admin/form_design/data/complete-form.js | 2 ++ 1 file changed, 2 insertions(+) diff --git a/src/openforms/js/components/admin/form_design/data/complete-form.js b/src/openforms/js/components/admin/form_design/data/complete-form.js index d1f2321be2..e5ebfd633e 100644 --- a/src/openforms/js/components/admin/form_design/data/complete-form.js +++ b/src/openforms/js/components/admin/form_design/data/complete-form.js @@ -121,6 +121,8 @@ const saveForm = async (state, csrftoken) => { form: {uuid}, } = state; const cleanedState = produce(state, draft => { + // ensure we don't overwrite the submission counter with a stale state + delete draft.form.submissionCounter; delete draft.form.registrationBackend; // deprecated delete draft.form.registrationBackendOptions; // deprecated normalizeLimit(draft, 'successfulSubmissionsRemovalLimit'); From cbc7608b9cd6922084f057c0cc424d6eca8a0541 Mon Sep 17 00:00:00 2001 From: Sergei Maertens Date: Thu, 2 Jan 2025 17:13:05 +0100 Subject: [PATCH 3/3] :fire: Remove dead/deprecated code The serializer fields were removed before in 08dfdbcec805b3f7f75d2048eb09f51d58b9e14c --- .../js/components/admin/form_design/data/complete-form.js | 2 -- 1 file changed, 2 deletions(-) diff --git a/src/openforms/js/components/admin/form_design/data/complete-form.js b/src/openforms/js/components/admin/form_design/data/complete-form.js index e5ebfd633e..ef7bdb793e 100644 --- a/src/openforms/js/components/admin/form_design/data/complete-form.js +++ b/src/openforms/js/components/admin/form_design/data/complete-form.js @@ -123,8 +123,6 @@ const saveForm = async (state, csrftoken) => { const cleanedState = produce(state, draft => { // ensure we don't overwrite the submission counter with a stale state delete draft.form.submissionCounter; - delete draft.form.registrationBackend; // deprecated - delete draft.form.registrationBackendOptions; // deprecated normalizeLimit(draft, 'successfulSubmissionsRemovalLimit'); normalizeLimit(draft, 'incompleteSubmissionsRemovalLimit'); normalizeLimit(draft, 'erroredSubmissionsRemovalLimit');