Skip to content

Commit

Permalink
update reply journo first and last name during sync
Browse files Browse the repository at this point in the history
  • Loading branch information
Allie Crevier committed Sep 10, 2020
1 parent b8c6c31 commit e7fb7b3
Show file tree
Hide file tree
Showing 5 changed files with 61 additions and 50 deletions.
6 changes: 3 additions & 3 deletions securedrop_client/api_jobs/sync.py
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@
from sqlalchemy.orm.session import Session

from securedrop_client.api_jobs.base import ApiJob
from securedrop_client.storage import get_remote_data, update_and_get_user, update_local_storage
from securedrop_client.storage import create_or_update_user, get_remote_data, update_local_storage

logger = logging.getLogger(__name__)

Expand Down Expand Up @@ -42,6 +42,6 @@ def call_api(self, api_client: API, session: Session) -> Any:
update_local_storage(session, sources, submissions, replies, self.data_dir)
user = api_client.get_current_user()
if "uuid" in user and "username" in user and "first_name" in user and "last_name" in user:
update_and_get_user(
user["uuid"], user["username"], user["first_name"], user["last_name"], session,
create_or_update_user(
user["uuid"], user["username"], user["first_name"], user["last_name"], session
)
6 changes: 3 additions & 3 deletions securedrop_client/logic.py
Original file line number Diff line number Diff line change
Expand Up @@ -457,11 +457,11 @@ def on_authenticate_success(self, result):
"""
logger.info("{} successfully logged in".format(self.api.username))
self.gui.hide_login()
user = storage.update_and_get_user(
user = storage.create_or_update_user(
self.api.token_journalist_uuid,
self.api.username,
self.api.journalist_first_name,
self.api.journalist_last_name,
self.api.first_name,
self.api.last_name,
self.session,
)
# Clear clipboard contents in case of previously pasted creds
Expand Down
47 changes: 27 additions & 20 deletions securedrop_client/storage.py
Original file line number Diff line number Diff line change
Expand Up @@ -299,8 +299,28 @@ def update_replies(
source_cache = SourceCache(session)
for reply in remote_replies:
user = users.get(reply.journalist_uuid)

if not user:
user = find_or_create_user(reply.journalist_uuid, reply.journalist_username, session)
user = create_or_update_user(
reply.journalist_uuid,
reply.journalist_username,
reply.journalist_first_name,
reply.journalist_last_name,
session,
)
users[reply.journalist_uuid] = user
elif (
user.username != reply.journalist_username
or user.firstname != reply.journalist_first_name
or user.lastname != reply.journalist_last_name
):
user = create_or_update_user(
reply.journalist_uuid,
reply.journalist_username,
reply.journalist_first_name,
reply.journalist_last_name,
session,
)
users[reply.journalist_uuid] = user

local_reply = local_replies_by_uuid.get(reply.uuid)
Expand Down Expand Up @@ -357,7 +377,9 @@ def update_replies(
session.commit()


def find_or_create_user(uuid: str, username: str, session: Session, commit: bool = True) -> User:
def create_or_update_user(
uuid: str, username: str, firstname: str, lastname: str, session: Session
) -> User:
"""
Returns a user object representing the referenced journalist UUID.
If the user does not already exist in the data, a new instance is created.
Expand All @@ -366,30 +388,15 @@ def find_or_create_user(uuid: str, username: str, session: Session, commit: bool
user = session.query(User).filter_by(uuid=uuid).one_or_none()

if not user:
new_user = User(username=username)
new_user = User(username=username, firstname=firstname, lastname=lastname)
new_user.uuid = uuid
session.add(new_user)
if commit:
session.commit()
session.commit()
return new_user

if user.username != username:
user.username = username
if commit:
session.commit()

return user


def update_and_get_user(
uuid: str, username: str, firstname: str, lastname: str, session: Session
) -> User:
"""
Returns a user object representing the referenced journalist UUID.
If user fields have changed, the db is updated.
"""
user = find_or_create_user(uuid, username, session)

session.commit()
if user.firstname != firstname:
user.firstname = firstname
session.commit()
Expand Down
4 changes: 2 additions & 2 deletions tests/test_logic.py
Original file line number Diff line number Diff line change
Expand Up @@ -219,8 +219,8 @@ def test_Controller_on_authenticate_success(homedir, config, mocker, session_mak
co.api = mocker.MagicMock()
co.api.token_journalist_uuid = user.uuid
co.api.username = user.username
co.api.journalist_first_name = user.firstname
co.api.journalist_last_name = user.lastname
co.api.first_name = user.firstname
co.api.last_name = user.lastname
co.resume_queues = mocker.MagicMock()

co.on_authenticate_success(True)
Expand Down
48 changes: 26 additions & 22 deletions tests/test_storage.py
Original file line number Diff line number Diff line change
Expand Up @@ -15,12 +15,12 @@
import securedrop_client.db
from securedrop_client import db
from securedrop_client.storage import (
create_or_update_user,
delete_local_source_by_uuid,
delete_single_submission_or_reply_on_disk,
find_new_files,
find_new_messages,
find_new_replies,
find_or_create_user,
get_file,
get_local_files,
get_local_messages,
Expand All @@ -35,7 +35,6 @@
mark_as_not_downloaded,
set_message_or_reply_content,
source_exists,
update_and_get_user,
update_draft_replies,
update_file_size,
update_files,
Expand Down Expand Up @@ -95,6 +94,8 @@ def make_remote_reply(source_uuid, journalist_uuid="testymctestface"):
filename="1-reply.filename",
journalist_uuid=journalist_uuid,
journalist_username="test",
journalist_first_name="test",
journalist_last_name="test",
file_counter=1,
is_deleted_by_source=False,
reply_url="test",
Expand Down Expand Up @@ -683,7 +684,7 @@ def test_update_messages(homedir, mocker):
local_user.id = 42
mock_session.query().filter_by().first.return_value = local_source
mock_focu = mocker.MagicMock(return_value=local_user)
mocker.patch("securedrop_client.storage.find_or_create_user", mock_focu)
mocker.patch("securedrop_client.storage.create_or_update_user", mock_focu)
mock_delete_submission_files = mocker.patch(
"securedrop_client.storage.delete_single_submission_or_reply_on_disk"
)
Expand Down Expand Up @@ -749,8 +750,10 @@ def test_update_replies(homedir, mocker, session):
# local database, the other will NOT exist in the local database
# (this will be added to the database)
remote_reply_update = factory.RemoteReply(
journalist_uuid=journalist.uuid,
uuid=local_reply_update.uuid,
journalist_uuid=journalist.uuid,
journalist_first_name="new_name",
journalist_last_name="new_name",
source_url="/api/v1/sources/{}".format(source.uuid),
file_counter=local_reply_update.file_counter,
filename=local_reply_update.filename,
Expand All @@ -761,6 +764,8 @@ def test_update_replies(homedir, mocker, session):
source_url="/api/v1/sources/{}".format(source.uuid),
file_counter=factory.REPLY_COUNT + 1,
filename="{}-filename.gpg".format(factory.REPLY_COUNT + 1),
journalist_first_name="",
journalist_last_name="",
)

remote_replies = [remote_reply_update, remote_reply_create]
Expand Down Expand Up @@ -867,70 +872,69 @@ def test_update_replies_missing_source(homedir, mocker, session):
error_logger.assert_called_once_with(f"No source found for reply {remote_reply.uuid}")


def test_find_or_create_user_existing_uuid(mocker):
def test_create_or_update_user_existing_uuid(mocker):
"""
Return an existing user object with the referenced uuid.
"""
mock_session = mocker.MagicMock()
mock_user = mocker.MagicMock()
mock_user.username = "foobar"
mock_user.firstname = "foobar"
mock_user.lastname = "foobar"
mock_session.query().filter_by().one_or_none.return_value = mock_user
assert find_or_create_user("uuid", "foobar", mock_session) == mock_user
assert create_or_update_user("uuid", "username", "fn", "ln", mock_session) == mock_user


def test_find_or_create_user_update_username(mocker, session):
def test_create_or_update_user_update_username(mocker, session):
"""
Return an existing user object with the updated username.
"""
user = factory.User(uuid="mock_uuid", username="mock_old_username")
session.add(user)

actual_user = find_or_create_user("mock_uuid", "mock_username", session)
actual_user = create_or_update_user("mock_uuid", "mock_username", "fn", "ln", session)

assert actual_user == user
assert actual_user.username == "mock_username"
assert actual_user.firstname == "fn"
assert actual_user.lastname == "ln"


def test_find_or_create_user_new(mocker):
def test_create_or_update_user_new(mocker):
"""
Create and return a user object for an unknown username.
"""
mock_session = mocker.MagicMock()
mock_session.query().filter_by().one_or_none.return_value = None
new_user = find_or_create_user("uuid", "unknown", mock_session)
new_user = create_or_update_user("uuid", "unknown", "unknown", "unknown", mock_session)
assert new_user.username == "unknown"
mock_session.add.assert_called_once_with(new_user)
mock_session.commit.assert_called_once_with()


def test_update_and_get_user(mocker, session):
def test_create_or_update_user(mocker, session):
"""
Return an existing user object with the updated username.
Return an existing user object with the updated username, firstname, and lastname.
"""
user = factory.User(
uuid="mock_uuid",
username="mock_username", # username is needed for find_or_create_user
username="mock_old_username", # username is needed for create_or_update_user
firstname="mock_old_firstname",
lastname="mock_old_lastname",
)
session.add(user)
find_or_create_user_fn = mocker.patch(
"securedrop_client.storage.find_or_create_user", return_value=user
)

actual_user = update_and_get_user(
updated_user = create_or_update_user(
uuid="mock_uuid",
username="mock_username",
firstname="mock_firstname",
lastname="mock_lastname",
session=session,
)

find_or_create_user_fn.assert_called_with("mock_uuid", "mock_username", session)
assert actual_user == user
assert actual_user.username == "mock_username"
assert actual_user.firstname == "mock_firstname"
assert actual_user.lastname == "mock_lastname"
assert updated_user.username == "mock_username"
assert updated_user.firstname == "mock_firstname"
assert updated_user.lastname == "mock_lastname"


def test_find_new_messages(mocker, session):
Expand Down

0 comments on commit e7fb7b3

Please sign in to comment.