Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add move to object client + change datasets.mv #163

Open
wants to merge 5 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
31 changes: 31 additions & 0 deletions faculty/clients/object.py
Original file line number Diff line number Diff line change
Expand Up @@ -239,6 +239,37 @@ def copy(self, project_id, source, destination, recursive=False):
else:
raise

def move(self, project_id, source, destination):
"""Move objects in the store

Parameters
----------
project_id : uuid.UUID
The project to move objects in.
source : str
Move object from this source path.
destination : str
Move object to this destination path.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Nit: remove extra blank line


Raises
------
PathNotFound
When the source path does not exist or is not found.
"""
url_encoded_destination = urllib.parse.quote(destination.lstrip("/"))

endpoint = "/project/{}/object-move/{}".format(
project_id, url_encoded_destination
)
params = {"sourcePath": source}

try:
self._put_raw(endpoint, params=params)
except NotFound as err:
if err.error_code == "source_path_not_found":
raise PathNotFound(source)
Comment on lines +270 to +271
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This should re-raise the original exception if the error code does not match:

Suggested change
if err.error_code == "source_path_not_found":
raise PathNotFound(source)
if err.error_code == "source_path_not_found":
raise PathNotFound(source)
else:
raise


def delete(self, project_id, path, recursive=False):
"""Delete objects in the store.

Expand Down
14 changes: 1 addition & 13 deletions faculty/datasets/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -334,19 +334,7 @@ def mv(source_path, destination_path, project_id=None, object_client=None):
project_id = project_id or get_context().project_id
object_client = object_client or ObjectClient(get_session())

cp(
source_path,
destination_path,
project_id=project_id,
recursive=True,
object_client=object_client,
)
rm(
source_path,
project_id=project_id,
recursive=True,
object_client=object_client,
)
object_client.move(project_id, source_path, destination_path)


def cp(
Expand Down
22 changes: 22 additions & 0 deletions tests/clients/test_object.py
Original file line number Diff line number Diff line change
Expand Up @@ -378,6 +378,28 @@ def test_object_client_copy_source_is_a_directory(mocker):
client.copy(PROJECT_ID, "source", "destination")


def test_object_client_move_url_encoding(mocker):
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Since this also functions as the main test of ObjectClient.move, I suggest naming this:

Suggested change
def test_object_client_move_url_encoding(mocker):
def test_object_client_move(mocker):

mocker.patch.object(ObjectClient, "_put_raw")

client = ObjectClient(mocker.Mock())
client.move(PROJECT_ID, "source", "/[1]/")

ObjectClient._put_raw.assert_called_once_with(
"/project/{}/object-move/%5B1%5D/".format(PROJECT_ID),
params={"sourcePath": "source"},
)


def test_object_client_move_source_not_found(mocker):
error_code = "source_path_not_found"
exception = NotFound(mocker.Mock(), mocker.Mock(), error_code)
mocker.patch.object(ObjectClient, "_put_raw", side_effect=exception)

client = ObjectClient(mocker.Mock())
with pytest.raises(PathNotFound, match="'source' cannot be found"):
client.move(PROJECT_ID, "source", "destination")


def test_object_client_delete_default(mocker):
path = "test-path"
mocker.patch.object(ObjectClient, "_delete_raw")
Expand Down
18 changes: 2 additions & 16 deletions tests/datasets/test_init.py
Original file line number Diff line number Diff line change
Expand Up @@ -419,23 +419,9 @@ def test_rmdir_nonempty_directory(mocker, prefix):


def test_mv(mocker, mock_client):
cp_mock = mocker.patch("faculty.datasets.cp")
rm_mock = mocker.patch("faculty.datasets.rm")

datasets.mv("source-path", "destination-path", project_id=PROJECT_ID)

cp_mock.assert_called_once_with(
"source-path",
"destination-path",
project_id=PROJECT_ID,
recursive=True,
object_client=mock_client,
)
rm_mock.assert_called_once_with(
"source-path",
project_id=PROJECT_ID,
recursive=True,
object_client=mock_client,
mock_client.move.assert_called_once_with(
PROJECT_ID, "source-path", "destination-path"
)


Expand Down