Skip to content

Commit

Permalink
[WIP] fix: don't override DraftReply.send_status based on application…
Browse files Browse the repository at this point in the history
…-level 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.

[WIP] refactor: consolidate mark_all_pending_drafts_as_failed() calls

1. when queue is cleared
2. when local storage updates after sync
  • Loading branch information
cfm committed May 25, 2022
1 parent 21e6cd2 commit 965a988
Show file tree
Hide file tree
Showing 2 changed files with 16 additions and 9 deletions.
22 changes: 16 additions & 6 deletions securedrop_client/logic.py
Original file line number Diff line number Diff line change
Expand Up @@ -361,6 +361,7 @@ def __init__( # type: ignore [no-untyped-def]

# Queue that handles running API job
self.api_job_queue = ApiJobQueue(self.api, self.session_maker)
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 @@ -470,6 +471,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 @@ -509,7 +513,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 @@ -579,7 +582,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 @@ -633,6 +635,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 @@ -775,10 +778,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 @@ -1170,3 +1169,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 @@ -1047,9 +1047,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

0 comments on commit 965a988

Please sign in to comment.