2023-04-10 22:17:12 -04:00
|
|
|
import pytest
|
2023-05-08 18:49:36 -04:00
|
|
|
from os.path import basename
|
2023-04-10 22:17:12 -04:00
|
|
|
from io import BytesIO
|
|
|
|
from werkzeug.datastructures import FileStorage
|
2023-05-17 18:40:31 -04:00
|
|
|
from sachet.server.models import Upload, Chunk
|
2023-05-08 18:49:36 -04:00
|
|
|
from sachet.server import storage
|
2023-04-15 16:33:50 -04:00
|
|
|
import uuid
|
2023-04-10 22:17:12 -04:00
|
|
|
|
|
|
|
"""Test file share endpoints."""
|
|
|
|
|
|
|
|
|
|
|
|
# if other storage backends are implemented we test them with the same suite
|
|
|
|
# this might be redundant because test_storage tests the backends already
|
|
|
|
@pytest.mark.parametrize("client", [{"SACHET_STORAGE": "filesystem"}], indirect=True)
|
|
|
|
class TestSuite:
|
2023-05-07 21:08:45 -04:00
|
|
|
def test_sharing(self, client, users, auth, rand, upload):
|
2023-04-10 22:17:12 -04:00
|
|
|
# create share
|
2023-04-15 17:56:48 -04:00
|
|
|
resp = client.post(
|
|
|
|
"/files", headers=auth("jeff"), json={"file_name": "content.bin"}
|
|
|
|
)
|
2023-04-10 22:17:12 -04:00
|
|
|
assert resp.status_code == 201
|
|
|
|
|
|
|
|
data = resp.get_json()
|
|
|
|
url = data.get("url")
|
|
|
|
|
|
|
|
assert url is not None
|
|
|
|
assert "/files/" in url
|
|
|
|
|
|
|
|
upload_data = rand.randbytes(4000)
|
|
|
|
|
2023-05-07 21:08:45 -04:00
|
|
|
resp = upload(
|
2023-04-10 22:17:12 -04:00
|
|
|
url + "/content",
|
2023-05-07 21:08:45 -04:00
|
|
|
BytesIO(upload_data),
|
2023-04-13 13:30:53 -04:00
|
|
|
headers=auth("jeff"),
|
2023-05-07 21:08:45 -04:00
|
|
|
chunk_size=1230,
|
2023-04-10 22:17:12 -04:00
|
|
|
)
|
|
|
|
assert resp.status_code == 201
|
|
|
|
|
|
|
|
# read file
|
|
|
|
resp = client.get(
|
|
|
|
url + "/content",
|
2023-04-13 13:30:53 -04:00
|
|
|
headers=auth("jeff"),
|
2023-04-10 22:17:12 -04:00
|
|
|
)
|
|
|
|
assert resp.data == upload_data
|
2023-04-15 18:12:50 -04:00
|
|
|
assert "filename=content.bin" in resp.headers["Content-Disposition"].split("; ")
|
2023-04-10 22:17:12 -04:00
|
|
|
|
|
|
|
# test deletion
|
|
|
|
resp = client.delete(
|
|
|
|
url,
|
2023-04-13 13:30:53 -04:00
|
|
|
headers=auth("jeff"),
|
2023-04-10 22:17:12 -04:00
|
|
|
)
|
|
|
|
assert resp.status_code == 200
|
|
|
|
|
|
|
|
# file shouldn't exist anymore
|
|
|
|
resp = client.get(
|
|
|
|
url + "/content",
|
2023-04-13 13:30:53 -04:00
|
|
|
headers=auth("jeff"),
|
2023-04-10 22:17:12 -04:00
|
|
|
)
|
|
|
|
assert resp.status_code == 404
|
2023-04-15 16:33:50 -04:00
|
|
|
|
2023-05-08 18:49:36 -04:00
|
|
|
for f in storage.list_files():
|
|
|
|
assert basename(url) not in f.name
|
|
|
|
|
2023-05-07 21:08:45 -04:00
|
|
|
def test_modification(self, client, users, auth, rand, upload):
|
2023-04-22 18:25:10 -04:00
|
|
|
# create share
|
|
|
|
resp = client.post(
|
|
|
|
"/files", headers=auth("jeff"), json={"file_name": "content.bin"}
|
|
|
|
)
|
|
|
|
data = resp.get_json()
|
|
|
|
url = data.get("url")
|
|
|
|
|
|
|
|
upload_data = rand.randbytes(4000)
|
|
|
|
new_data = rand.randbytes(4000)
|
|
|
|
|
|
|
|
# upload file to share
|
2023-05-07 21:08:45 -04:00
|
|
|
resp = upload(
|
2023-04-22 18:25:10 -04:00
|
|
|
url + "/content",
|
2023-05-07 21:08:45 -04:00
|
|
|
BytesIO(upload_data),
|
2023-04-22 18:25:10 -04:00
|
|
|
headers=auth("jeff"),
|
|
|
|
)
|
2023-05-07 21:08:45 -04:00
|
|
|
assert resp.status_code == 201
|
2023-04-22 18:25:10 -04:00
|
|
|
|
|
|
|
# modify metadata
|
|
|
|
resp = client.patch(
|
|
|
|
url,
|
|
|
|
headers=auth("jeff"),
|
|
|
|
json={"file_name": "new_bin.bin"},
|
|
|
|
)
|
|
|
|
assert resp.status_code == 200
|
|
|
|
|
|
|
|
# modify file contents
|
2023-05-07 21:08:45 -04:00
|
|
|
resp = upload(
|
2023-04-22 18:25:10 -04:00
|
|
|
url + "/content",
|
2023-05-07 21:08:45 -04:00
|
|
|
BytesIO(new_data),
|
2023-04-22 18:25:10 -04:00
|
|
|
headers=auth("jeff"),
|
2023-05-07 21:08:45 -04:00
|
|
|
method=client.put,
|
2023-04-22 18:25:10 -04:00
|
|
|
)
|
2023-05-07 21:08:45 -04:00
|
|
|
assert resp.status_code == 201
|
2023-04-22 18:25:10 -04:00
|
|
|
|
|
|
|
# read file
|
|
|
|
resp = client.get(
|
|
|
|
url + "/content",
|
|
|
|
headers=auth("jeff"),
|
|
|
|
)
|
|
|
|
assert resp.data == new_data
|
|
|
|
assert "filename=new_bin.bin" in resp.headers["Content-Disposition"].split("; ")
|
|
|
|
|
2023-05-21 22:18:45 -04:00
|
|
|
def test_transfer(self, client, users, auth):
|
|
|
|
# create share
|
|
|
|
resp = client.post(
|
|
|
|
"/files", headers=auth("jeff"), json={"file_name": "content.bin"}
|
|
|
|
)
|
|
|
|
data = resp.get_json()
|
|
|
|
url = data.get("url")
|
|
|
|
|
|
|
|
# transfer ownership over to dave
|
|
|
|
resp = client.patch(url, headers=auth("jeff"), json={"owner_name": "dave"})
|
|
|
|
assert resp.status_code == 200
|
|
|
|
|
|
|
|
# ensure the transfer worked
|
|
|
|
resp = client.patch(
|
|
|
|
url, headers=auth("jeff"), json={"file_name": "jeff's file"}
|
|
|
|
)
|
|
|
|
assert resp.status_code == 403
|
|
|
|
resp = client.patch(
|
|
|
|
url, headers=auth("dave"), json={"file_name": "dave's file"}
|
|
|
|
)
|
|
|
|
assert resp.status_code == 200
|
|
|
|
|
|
|
|
# transfer ownership back to jeff
|
|
|
|
resp = client.patch(url, headers=auth("dave"), json={"owner_name": "jeff"})
|
|
|
|
assert resp.status_code == 200
|
|
|
|
|
|
|
|
# ensure the transfer worked
|
|
|
|
resp = client.patch(
|
|
|
|
url, headers=auth("dave"), json={"file_name": "dave's epic file"}
|
|
|
|
)
|
|
|
|
assert resp.status_code == 403
|
|
|
|
resp = client.patch(
|
|
|
|
url, headers=auth("jeff"), json={"file_name": "jeff's file"}
|
|
|
|
)
|
|
|
|
assert resp.status_code == 200
|
|
|
|
|
2023-05-07 21:08:45 -04:00
|
|
|
def test_invalid(self, client, users, auth, rand, upload):
|
2023-04-15 16:33:50 -04:00
|
|
|
"""Test invalid requests."""
|
|
|
|
|
|
|
|
upload_data = rand.randbytes(4000)
|
|
|
|
|
|
|
|
# unauthenticated
|
|
|
|
resp = client.post("/files")
|
|
|
|
assert resp.status_code == 401
|
|
|
|
|
|
|
|
# non-existent
|
|
|
|
resp = client.get("/files/" + str(uuid.UUID(int=0)), headers=auth("jeff"))
|
|
|
|
assert resp.status_code == 404
|
|
|
|
resp = client.get(
|
|
|
|
"/files/" + str(uuid.UUID(int=0)) + "/content", headers=auth("jeff")
|
|
|
|
)
|
|
|
|
assert resp.status_code == 404
|
|
|
|
resp = client.post(
|
|
|
|
"/files/" + str(uuid.UUID(int=0)) + "/content", headers=auth("jeff")
|
|
|
|
)
|
|
|
|
assert resp.status_code == 404
|
|
|
|
resp = client.put(
|
|
|
|
"/files/" + str(uuid.UUID(int=0)) + "/content", headers=auth("jeff")
|
|
|
|
)
|
|
|
|
assert resp.status_code == 404
|
2023-06-25 20:41:43 -04:00
|
|
|
resp = client.delete("/files/" + str(uuid.UUID(int=0)), headers=auth("jeff"))
|
|
|
|
assert resp.status_code == 404
|
|
|
|
resp = client.patch(
|
|
|
|
"/files/" + str(uuid.UUID(int=0)),
|
|
|
|
headers=auth("jeff"),
|
|
|
|
json=dict(filename="incredible-new-filename"),
|
|
|
|
)
|
|
|
|
assert resp.status_code == 404
|
|
|
|
resp = client.put(
|
|
|
|
"/files/" + str(uuid.UUID(int=0)),
|
|
|
|
headers=auth("jeff"),
|
|
|
|
json=dict(filename="incredible-new-filename", owner_name="jeff"),
|
|
|
|
)
|
|
|
|
assert resp.status_code == 404
|
2023-04-15 16:33:50 -04:00
|
|
|
|
2023-07-18 13:26:48 -04:00
|
|
|
# malformed
|
|
|
|
resp = client.get("/files/" + "123123123123", headers=auth("jeff"))
|
|
|
|
assert resp.status_code == 404
|
|
|
|
resp = client.get("/files/" + "123123123" + "/content", headers=auth("jeff"))
|
|
|
|
assert resp.status_code == 404
|
|
|
|
resp = client.post("/files/" + "123123123" + "/content", headers=auth("jeff"))
|
|
|
|
assert resp.status_code == 404
|
|
|
|
resp = client.put("/files/" + "123123123" + "/content", headers=auth("jeff"))
|
|
|
|
assert resp.status_code == 404
|
|
|
|
resp = client.delete("/files/" + "123123123", headers=auth("jeff"))
|
|
|
|
assert resp.status_code == 404
|
|
|
|
resp = client.patch(
|
|
|
|
"/files/" + "123123123",
|
|
|
|
headers=auth("jeff"),
|
|
|
|
json=dict(filename="incredible-new-filename"),
|
|
|
|
)
|
|
|
|
assert resp.status_code == 404
|
|
|
|
resp = client.put(
|
|
|
|
"/files/" + "123123123",
|
|
|
|
headers=auth("jeff"),
|
|
|
|
json=dict(filename="incredible-new-filename", owner_name="jeff"),
|
|
|
|
)
|
|
|
|
assert resp.status_code == 404
|
|
|
|
|
2023-04-15 16:33:50 -04:00
|
|
|
# no CREATE permission
|
|
|
|
resp = client.post("/files", headers=auth("no_create_user"))
|
|
|
|
assert resp.status_code == 403
|
|
|
|
|
|
|
|
# valid share creation to move on to testing content endpoint
|
2023-04-15 17:56:48 -04:00
|
|
|
resp = client.post(
|
|
|
|
"/files", headers=auth("jeff"), json={"file_name": "content.bin"}
|
|
|
|
)
|
2023-04-15 16:33:50 -04:00
|
|
|
assert resp.status_code == 201
|
|
|
|
data = resp.get_json()
|
|
|
|
url = data.get("url")
|
|
|
|
|
|
|
|
# test invalid methods
|
2023-05-07 21:08:45 -04:00
|
|
|
resp = upload(
|
2023-04-15 16:33:50 -04:00
|
|
|
url + "/content",
|
2023-05-07 21:08:45 -04:00
|
|
|
BytesIO(upload_data),
|
2023-04-15 16:33:50 -04:00
|
|
|
headers=auth("jeff"),
|
2023-05-07 21:08:45 -04:00
|
|
|
method=client.put,
|
2023-04-15 16:33:50 -04:00
|
|
|
)
|
|
|
|
assert resp.status_code == 423
|
2023-05-07 21:08:45 -04:00
|
|
|
resp = upload(
|
2023-04-15 16:33:50 -04:00
|
|
|
url + "/content",
|
2023-05-07 21:08:45 -04:00
|
|
|
BytesIO(upload_data),
|
2023-04-15 16:33:50 -04:00
|
|
|
headers=auth("jeff"),
|
2023-05-07 21:08:45 -04:00
|
|
|
method=client.patch,
|
2023-04-15 16:33:50 -04:00
|
|
|
)
|
|
|
|
assert resp.status_code == 405
|
|
|
|
|
|
|
|
# test other user being unable to upload to this share
|
2023-05-07 21:08:45 -04:00
|
|
|
resp = upload(
|
2023-04-15 16:33:50 -04:00
|
|
|
url + "/content",
|
2023-05-07 21:08:45 -04:00
|
|
|
BytesIO(upload_data),
|
2023-04-15 16:33:50 -04:00
|
|
|
headers=auth("dave"),
|
|
|
|
)
|
|
|
|
assert resp.status_code == 403
|
|
|
|
|
|
|
|
# upload file to share (properly)
|
2023-05-07 21:08:45 -04:00
|
|
|
resp = upload(
|
2023-04-15 16:33:50 -04:00
|
|
|
url + "/content",
|
2023-05-07 21:08:45 -04:00
|
|
|
BytesIO(upload_data),
|
2023-04-15 16:33:50 -04:00
|
|
|
headers=auth("jeff"),
|
|
|
|
)
|
|
|
|
assert resp.status_code == 201
|
|
|
|
|
2023-04-29 12:09:32 -04:00
|
|
|
# test other user being unable to modify this share
|
2023-05-07 21:08:45 -04:00
|
|
|
resp = upload(
|
2023-04-29 12:09:32 -04:00
|
|
|
url + "/content",
|
2023-05-07 21:08:45 -04:00
|
|
|
BytesIO(upload_data),
|
2023-04-29 12:09:32 -04:00
|
|
|
headers=auth("dave"),
|
2023-05-07 21:08:45 -04:00
|
|
|
method=client.put,
|
2023-04-29 12:09:32 -04:00
|
|
|
)
|
|
|
|
assert resp.status_code == 403
|
2023-05-21 22:01:02 -04:00
|
|
|
resp = client.patch(
|
2023-05-21 22:18:45 -04:00
|
|
|
url, headers=auth("dave"), json=dict(file_name="epic_new_filename.bin")
|
2023-05-21 22:01:02 -04:00
|
|
|
)
|
|
|
|
assert resp.status_code == 403
|
|
|
|
resp = client.put(
|
|
|
|
url,
|
|
|
|
headers=auth("dave"),
|
2023-05-21 22:18:45 -04:00
|
|
|
json=dict(file_name="epic_new_filename.bin", owner_name="dave"),
|
2023-05-21 22:01:02 -04:00
|
|
|
)
|
|
|
|
assert resp.status_code == 403
|
2023-04-29 12:09:32 -04:00
|
|
|
|
2023-05-21 22:18:45 -04:00
|
|
|
# test assigning a file to a non-existent user
|
|
|
|
resp = client.patch(
|
|
|
|
url, headers=auth("jeff"), json=dict(owner_name="non_existent_user")
|
|
|
|
)
|
|
|
|
assert resp.status_code == 400
|
|
|
|
|
2023-04-15 16:33:50 -04:00
|
|
|
# test not allowing re-upload
|
2023-05-07 21:08:45 -04:00
|
|
|
resp = upload(
|
2023-04-15 16:33:50 -04:00
|
|
|
url + "/content",
|
2023-05-07 21:08:45 -04:00
|
|
|
BytesIO(upload_data),
|
2023-04-15 16:33:50 -04:00
|
|
|
headers=auth("jeff"),
|
|
|
|
)
|
|
|
|
assert resp.status_code == 423
|
|
|
|
|
|
|
|
# no READ permission
|
|
|
|
resp = client.get(url, headers=auth("no_read_user"))
|
|
|
|
assert resp.status_code == 403
|
|
|
|
resp = client.get(url + "/content", headers=auth("no_read_user"))
|
|
|
|
assert resp.status_code == 403
|
2023-04-29 13:18:09 -04:00
|
|
|
|
2023-05-07 21:08:45 -04:00
|
|
|
def test_locking(self, client, users, auth, rand, upload):
|
2023-04-29 13:18:09 -04:00
|
|
|
# upload share
|
|
|
|
resp = client.post(
|
|
|
|
"/files", headers=auth("jeff"), json={"file_name": "content.bin"}
|
|
|
|
)
|
|
|
|
data = resp.get_json()
|
|
|
|
url = data.get("url")
|
|
|
|
upload_data = rand.randbytes(4000)
|
2023-05-07 21:08:45 -04:00
|
|
|
resp = upload(
|
2023-04-29 13:18:09 -04:00
|
|
|
url + "/content",
|
2023-05-07 21:08:45 -04:00
|
|
|
BytesIO(upload_data),
|
2023-04-29 13:18:09 -04:00
|
|
|
headers=auth("jeff"),
|
|
|
|
)
|
|
|
|
assert resp.status_code == 201
|
|
|
|
|
|
|
|
# lock share
|
|
|
|
resp = client.post(
|
|
|
|
url + "/lock",
|
|
|
|
headers=auth("jeff"),
|
|
|
|
)
|
|
|
|
assert resp.status_code == 200
|
|
|
|
|
|
|
|
# attempt to modify share
|
2023-05-07 21:08:45 -04:00
|
|
|
resp = upload(
|
2023-04-29 13:18:09 -04:00
|
|
|
url + "/content",
|
2023-05-07 21:08:45 -04:00
|
|
|
BytesIO(upload_data),
|
2023-04-29 13:18:09 -04:00
|
|
|
headers=auth("jeff"),
|
2023-05-07 21:08:45 -04:00
|
|
|
method=client.put,
|
2023-04-29 13:18:09 -04:00
|
|
|
)
|
|
|
|
assert resp.status_code == 423
|
|
|
|
|
|
|
|
# attempt to delete share
|
|
|
|
resp = client.delete(
|
|
|
|
url,
|
|
|
|
headers=auth("jeff"),
|
|
|
|
)
|
|
|
|
assert resp.status_code == 423
|
|
|
|
|
|
|
|
# unlock share
|
|
|
|
resp = client.post(
|
|
|
|
url + "/unlock",
|
|
|
|
headers=auth("jeff"),
|
|
|
|
)
|
|
|
|
assert resp.status_code == 200
|
|
|
|
|
|
|
|
# attempt to modify share
|
2023-05-07 21:08:45 -04:00
|
|
|
resp = upload(
|
2023-04-29 13:18:09 -04:00
|
|
|
url + "/content",
|
2023-05-07 21:08:45 -04:00
|
|
|
BytesIO(upload_data),
|
2023-04-29 13:18:09 -04:00
|
|
|
headers=auth("jeff"),
|
2023-05-07 21:08:45 -04:00
|
|
|
method=client.put,
|
2023-04-29 13:18:09 -04:00
|
|
|
)
|
2023-05-07 21:08:45 -04:00
|
|
|
assert resp.status_code == 201
|
2023-04-29 13:18:09 -04:00
|
|
|
|
|
|
|
# attempt to delete share
|
|
|
|
resp = client.delete(
|
|
|
|
url,
|
|
|
|
headers=auth("jeff"),
|
|
|
|
)
|
|
|
|
assert resp.status_code == 200
|
|
|
|
|
|
|
|
# attempt to lock/unlock without perms
|
|
|
|
resp = client.post(
|
|
|
|
url + "/lock",
|
|
|
|
headers=auth("no_lock_user"),
|
|
|
|
)
|
|
|
|
assert resp.status_code == 403
|
|
|
|
resp = client.post(
|
|
|
|
url + "/unlock",
|
|
|
|
headers=auth("no_lock_user"),
|
|
|
|
)
|
|
|
|
assert resp.status_code == 403
|
2023-05-11 21:00:50 -04:00
|
|
|
|
|
|
|
def test_partial(self, client, users, auth, rand, upload):
|
|
|
|
# create share
|
|
|
|
resp = client.post(
|
|
|
|
"/files", headers=auth("jeff"), json={"file_name": "content.bin"}
|
|
|
|
)
|
|
|
|
assert resp.status_code == 201
|
|
|
|
|
|
|
|
data = resp.get_json()
|
|
|
|
url = data.get("url")
|
|
|
|
|
|
|
|
upload_data = b"1234567890" * 400
|
|
|
|
|
|
|
|
resp = upload(
|
|
|
|
url + "/content",
|
|
|
|
BytesIO(upload_data),
|
|
|
|
headers=auth("jeff"),
|
|
|
|
chunk_size=1230,
|
|
|
|
)
|
|
|
|
assert resp.status_code == 201
|
|
|
|
|
|
|
|
# test the following ranges
|
|
|
|
ranges = [
|
|
|
|
[0, 1],
|
|
|
|
[1, 1],
|
|
|
|
[2, 300],
|
|
|
|
[300, 30],
|
|
|
|
[3, 4],
|
|
|
|
[30, 3999],
|
|
|
|
[4000, 4000],
|
|
|
|
[3999, 39999],
|
|
|
|
[40000, 0],
|
|
|
|
[48000, 9],
|
|
|
|
[-1, 0],
|
|
|
|
[-2, 3],
|
|
|
|
[0, 4000],
|
|
|
|
[0, ""],
|
|
|
|
]
|
|
|
|
|
|
|
|
for r in ranges:
|
|
|
|
resp = client.get(
|
|
|
|
url + "/content",
|
|
|
|
headers=auth("jeff", data={"Range": f"bytes={r[0]}-{r[1]}"}),
|
|
|
|
)
|
|
|
|
if r[1] == "":
|
|
|
|
r[1] = len(upload_data)
|
|
|
|
# apparently if you specify an endpoint past the end
|
|
|
|
# it just truncates the response to the end
|
|
|
|
if r[0] < 0 or r[0] >= 4000:
|
|
|
|
assert resp.status_code == 416
|
|
|
|
elif r[0] > r[1]:
|
|
|
|
assert resp.status_code == 416
|
|
|
|
else:
|
|
|
|
assert resp.status_code == 206
|
|
|
|
assert resp.data == upload_data[r[0] : r[1] + 1]
|