Skip to content

Commit

Permalink
fix: don't override DraftReply.send_status based on application-level…
Browse files Browse the repository at this point in the history
… state

Since #750, application-level state transitions (logging in, logging
out, and switching into offline mode) call
securedrop_client.storage.mark_all_pending_draft_replies_as_failed().
However, one or more SendReplyJobs (and their underlying POST requests
to "/sources/<source_uuid/replies>") may be in flight on the network at
that time, whether or not the application is connected (or even running)
to receive their responses.  Until we have better (ideally generalized)
logic around upload jobs that have not yet been confirmed by the server,
these application-level events should not make assumptions about the
results of jobs that have already been dispatched.

Individual SendReplyJobs still call their own _set_status_to_failed()
method on non-timeout exceptions.
  • Loading branch information
cfm committed Nov 8, 2022
1 parent f70327c commit 87e0a99
Show file tree
Hide file tree
Showing 3 changed files with 65 additions and 22 deletions.
22 changes: 16 additions & 6 deletions securedrop_client/logic.py
Original file line number Diff line number Diff line change
Expand Up @@ -381,6 +381,7 @@ def __init__( # type: ignore [no-untyped-def]
self.api_job_queue = ApiJobQueue(
self.api, self.session_maker, self.main_queue_thread, self.file_download_queue_thread
)
self.api_job_queue.cleared.connect(self.on_queue_cleared)
self.api_job_queue.paused.connect(self.on_queue_paused)
self.add_job.connect(self.api_job_queue.enqueue)

Expand Down Expand Up @@ -481,6 +482,9 @@ def call_api( # type: ignore [no-untyped-def]
# Start the thread and related activity.
new_api_thread.start()

def on_queue_cleared(self) -> None:
self.update_failed_replies()

def on_queue_paused(self) -> None:
self.gui.update_error_status(
_("The SecureDrop server cannot be reached. Trying to reconnect..."), duration=0
Expand Down Expand Up @@ -526,7 +530,6 @@ def login(self, username: str, password: str, totp: str) -> None:
default_request_timeout for Queue API requests in ApiJobQueue in order to display errors
faster.
"""
storage.mark_all_pending_drafts_as_failed(self.session)
self.api = sdclientapi.API(
self.hostname, username, password, totp, self.proxy, default_request_timeout=60
)
Expand Down Expand Up @@ -596,7 +599,6 @@ def login_offline_mode(self) -> None:
# may have attempted online mode login, then switched to offline)
self.gui.clear_clipboard()
self.gui.show_main_window()
storage.mark_all_pending_drafts_as_failed(self.session)
self.update_sources()
self.show_last_sync()
self.show_last_sync_timer.start(TIME_BETWEEN_SHOWING_LAST_SYNC_MS)
Expand Down Expand Up @@ -650,6 +652,7 @@ def on_sync_success(self) -> None:
self.gui.refresh_current_source_conversation()
self.download_new_messages()
self.download_new_replies()
self.update_failed_replies()
self.sync_succeeded.emit()

try:
Expand Down Expand Up @@ -793,10 +796,6 @@ def logout(self) -> None:

self.invalidate_token()

failed_replies = storage.mark_all_pending_drafts_as_failed(self.session)
for failed_reply in failed_replies:
self.reply_failed.emit(failed_reply.uuid)

self.api_sync.stop()
self.api_job_queue.stop()
self.gui.logout()
Expand Down Expand Up @@ -1130,3 +1129,14 @@ def on_logout_success(self, result: Exception) -> None:

def on_logout_failure(self, result: Exception) -> None:
logging.info("Client logout failure")

def update_failed_replies(self) -> None:
"""
Emit an explicit `reply_failed` signal for each pending reply marked as failed by
the storage layer rather than the API job responsible for sending it (for example,
if the application quit mid-job or mid-queue). Without this signal, the reply
won't be shown as failed in the GUI until the application is restarted.
"""
failed_replies = storage.mark_all_pending_drafts_as_failed(self.session)
for failed_reply in failed_replies:
self.reply_failed.emit(failed_reply.uuid)
3 changes: 0 additions & 3 deletions securedrop_client/storage.py
Original file line number Diff line number Diff line change
Expand Up @@ -1079,9 +1079,6 @@ def get_reply(session: Session, uuid: str) -> Reply:


def mark_all_pending_drafts_as_failed(session: Session) -> List[DraftReply]:
"""
When we login (offline or online) or logout, we need to set all the pending replies as failed.
"""
pending_status = (
session.query(ReplySendStatus).filter_by(name=ReplySendStatusCodes.PENDING.value).one()
)
Expand Down
62 changes: 49 additions & 13 deletions tests/test_logic.py
Original file line number Diff line number Diff line change
Expand Up @@ -167,7 +167,6 @@ def test_Controller_login(homedir, config, mocker, session_maker):
"""
mock_gui = mocker.MagicMock()
mock_api = mocker.patch("securedrop_client.logic.sdclientapi.API")
fail_draft_replies = mocker.patch("securedrop_client.storage.mark_all_pending_drafts_as_failed")

co = Controller("http://localhost", mock_gui, session_maker, homedir, None)
co.call_api = mocker.MagicMock()
Expand All @@ -178,7 +177,6 @@ def test_Controller_login(homedir, config, mocker, session_maker):
co.call_api.assert_called_once_with(
mock_api().authenticate, co.on_authenticate_success, co.on_authenticate_failure
)
fail_draft_replies.assert_called_once_with(co.session)
co.show_last_sync_timer.stop.assert_called_once_with()


Expand Down Expand Up @@ -1031,7 +1029,8 @@ def test_Controller_invalidate_token(mocker, homedir, session_maker):

def test_Controller_logout_with_pending_replies(mocker, session_maker, homedir, reply_status_codes):
"""
Ensure draft reply fails on logout and that the reply_failed signal is emitted.
Ensure pending replies not currently being processed are marked failed and emit the
"reply_failed" signal.
"""
co = Controller("http://localhost", mocker.MagicMock(), session_maker, homedir, None)
co.api_job_queue = mocker.MagicMock()
Expand All @@ -1055,11 +1054,15 @@ def test_Controller_logout_with_pending_replies(mocker, session_maker, homedir,

co.logout()

for draft in session.query(db.DraftReply).all():
assert draft.send_status == failed_status
for (i, draft) in enumerate(session.query(db.DraftReply).all()):
if i == 0:
# We can't check that draft.sending_pid == os.getpid() here, because that's set by the
# SendReplyJob.
assert draft.send_status == pending_status
else:
assert draft.send_status == failed_status
co.reply_failed.emit.assert_called_once_with(draft.uuid)

assert len(reply_failed_emissions) == 1
assert reply_failed_emissions[0] == [pending_draft_reply.uuid]
co.api_job_queue.stop.assert_called_once_with()


Expand All @@ -1075,15 +1078,13 @@ def test_Controller_logout_with_no_api(homedir, config, mocker, session_maker):
co.api_job_queue = mocker.MagicMock()
co.api_job_queue.stop = mocker.MagicMock()
co.call_api = mocker.MagicMock()
fail_draft_replies = mocker.patch("securedrop_client.storage.mark_all_pending_drafts_as_failed")

co.logout()

assert not co.authenticated_user
co.call_api.assert_not_called()
co.api_job_queue.stop.assert_called_once_with()
co.gui.logout.assert_called_once_with()
fail_draft_replies.called_once_with(co.session)


def test_Controller_logout_success(homedir, config, mocker, session_maker):
Expand All @@ -1101,7 +1102,6 @@ def test_Controller_logout_success(homedir, config, mocker, session_maker):
co.call_api = mocker.MagicMock()
co.show_last_sync_timer = mocker.MagicMock()
info_logger = mocker.patch("securedrop_client.logic.logging.info")
fail_draft_replies = mocker.patch("securedrop_client.storage.mark_all_pending_drafts_as_failed")
logout_method = co.api.logout
co.logout()
co.call_api.assert_called_with(logout_method, co.on_logout_success, co.on_logout_failure)
Expand All @@ -1111,7 +1111,6 @@ def test_Controller_logout_success(homedir, config, mocker, session_maker):
co.gui.logout.assert_called_once_with()
msg = "Client logout successful"
info_logger.assert_called_once_with(msg)
fail_draft_replies.called_once_with(co.session)
co.show_last_sync_timer.start.assert_called_once_with(TIME_BETWEEN_SHOWING_LAST_SYNC_MS)


Expand All @@ -1129,7 +1128,6 @@ def test_Controller_logout_failure(homedir, config, mocker, session_maker):
co.api_job_queue.stop = mocker.MagicMock()
co.call_api = mocker.MagicMock()
info_logger = mocker.patch("securedrop_client.logic.logging.info")
fail_draft_replies = mocker.patch("securedrop_client.storage.mark_all_pending_drafts_as_failed")
logout_method = co.api.logout

co.logout()
Expand All @@ -1141,7 +1139,6 @@ def test_Controller_logout_failure(homedir, config, mocker, session_maker):
co.gui.logout.assert_called_once_with()
msg = "Client logout failure"
info_logger.assert_called_once_with(msg)
fail_draft_replies.called_once_with(co.session)


def test_Controller_set_activity_status(homedir, config, mocker, session_maker):
Expand Down Expand Up @@ -2331,6 +2328,21 @@ def test_APICallRunner_api_call_timeout(mocker, exception):
assert len(call_timed_out_emissions) == 1


def test_Controller_on_queue_cleared(homedir, config, mocker, session_maker):
"""
Check that clearing the queue marks as failed replies that were still pending in the queue.
"""
mock_gui = mocker.MagicMock()
co = Controller("http://localhost", mock_gui, session_maker, homedir, None)
fail_draft_replies = mocker.patch("securedrop_client.storage.mark_all_pending_drafts_as_failed")

mocker.patch.object(co, "api_job_queue")
co.api = "not none"
co.show_last_sync_timer = mocker.MagicMock()
co.on_queue_cleared()
fail_draft_replies.called_once_with(co.session)


def test_Controller_on_queue_paused(homedir, config, mocker, session_maker):
"""
Check that a paused queue is communicated to the user via the error status bar
Expand Down Expand Up @@ -2394,3 +2406,27 @@ def test_get_file(mocker, session, homedir):

storage.get_file.assert_called_once_with(co.session, file.uuid)
assert obj == file


def test_Controller_update_failed_replies(homedir, config, mocker, session, session_maker):
"""
The "reply_failed" signal is emitted for each pending reply marked as failed.
"""
mock_storage = mocker.patch("securedrop_client.logic.storage")
mock_storage.mark_all_pending_drafts_as_failed = mocker.MagicMock()

controller = Controller(
"http://localhost", mocker.MagicMock(), session_maker, homedir, None, None
)
controller.authenticated_user = factory.User(id=1)
controller.reply_failed = mocker.MagicMock()

source = factory.Source()
draft = factory.DraftReply(source=source, journalist_id=controller.authenticated_user.id)

failed_drafts = [draft]
mock_storage.mark_all_pending_drafts_as_failed.return_value = failed_drafts

controller.update_failed_replies()
for failed in failed_drafts:
controller.reply_failed.emit.assert_called_once_with(failed.uuid)

0 comments on commit 87e0a99

Please sign in to comment.