From 62c94a800174b62461843e5a7cc8856fb640036d Mon Sep 17 00:00:00 2001 From: redshiftzero Date: Thu, 27 Feb 2020 11:59:31 -0500 Subject: [PATCH 1/2] app: update source.interaction_count after successful reply --- securedrop_client/api_jobs/uploads.py | 2 ++ 1 file changed, 2 insertions(+) diff --git a/securedrop_client/api_jobs/uploads.py b/securedrop_client/api_jobs/uploads.py index 8616b7115..c45ca9f25 100644 --- a/securedrop_client/api_jobs/uploads.py +++ b/securedrop_client/api_jobs/uploads.py @@ -63,6 +63,8 @@ def call_api(self, api_client: API, session: Session) -> str: # Delete draft, add reply to replies table. session.add(reply_db_object) session.delete(draft_reply_db_object) + source.interaction_count += 1 + session.add(source) session.commit() return reply_db_object.uuid From fa9f4daae2ad13a3b01294fd5f42d7ca939effac Mon Sep 17 00:00:00 2001 From: redshiftzero Date: Thu, 27 Feb 2020 13:13:08 -0500 Subject: [PATCH 2/2] test: add regression test and explanation for #653 --- tests/api_jobs/test_uploads.py | 18 +++++++++++++++++- 1 file changed, 17 insertions(+), 1 deletion(-) diff --git a/tests/api_jobs/test_uploads.py b/tests/api_jobs/test_uploads.py index d607912cf..632d15f45 100644 --- a/tests/api_jobs/test_uploads.py +++ b/tests/api_jobs/test_uploads.py @@ -68,7 +68,10 @@ def test_drafts_ordering(homedir, mocker, session, session_maker, Check that if a reply is successful, drafts sent before and after continue to appear in the same order. ''' - source = factory.Source(interaction_count=1) + initial_interaction_count = 1 + source_uuid = 'foo' + source = factory.Source(uuid=source_uuid, + interaction_count=initial_interaction_count) session.add(source) msg_uuid = 'xyz456' @@ -130,6 +133,19 @@ def test_drafts_ordering(homedir, mocker, session, session_maker, reply = session.query(db.Reply).filter_by(uuid=msg_uuid).one() assert reply.journalist_id == api_client.token_journalist_uuid + # We use the file_counter on each Reply, Message, File, and DraftReply + # object to order the conversation view. We expect a unique file_counter + # for Reply, Message, and File objects since they are retrieved from + # the server. + # For DraftReply, we don't have that unique constraint, and we instead expect + # the file_counter to be the interaction_count at the time of send. + # If we do not update the interaction_count after each successful reply send, + # future drafts will have an interaction_count that is too low, leading + # to incorrectly ordered drafts until a metadata sync completes (the metadata + # sync is the place where the source object is updated from the server). + source = session.query(db.Source).filter_by(uuid=source_uuid).one() + assert source.interaction_count == initial_interaction_count + 1 + # Check the ordering displayed to the user assert source.collection[0] == draft_reply_before assert source.collection[1] == reply