import os from starlette.formparsers import UploadFile from starlette.requests import Request from starlette.responses import JSONResponse from starlette.testclient import TestClient class ForceMultipartDict(dict): def __bool__(self): return True # FORCE_MULTIPART is an empty dict that boolean-evaluates as `True`. FORCE_MULTIPART = ForceMultipartDict() def app(scope): async def asgi(receive, send): request = Request(scope, receive) data = await request.form() output = {} for key, value in data.items(): if isinstance(value, UploadFile): content = await value.read() output[key] = {"filename": value.filename, "content": content.decode()} else: output[key] = value await request.close() response = JSONResponse(output) await response(receive, send) return asgi def multi_items_app(scope): async def asgi(receive, send): request = Request(scope, receive) data = await request.form() output = {} for key, value in data.multi_items(): if key not in output: output[key] = [] if isinstance(value, UploadFile): content = await value.read() output[key].append( {"filename": value.filename, "content": content.decode()} ) else: output[key].append(value) await request.close() response = JSONResponse(output) await response(receive, send) return asgi def app_read_body(scope): async def asgi(receive, send): request = Request(scope, receive) # Read bytes, to force request.stream() to return the already parsed body body_bytes = await request.body() data = await request.form() output = {} for key, value in data.items(): output[key] = value await request.close() response = JSONResponse(output) await response(receive, send) return asgi def test_multipart_request_data(tmpdir): client = TestClient(app) response = client.post("/", data={"some": "data"}, files=FORCE_MULTIPART) assert response.json() == {"some": "data"} def test_multipart_request_files(tmpdir): path = os.path.join(tmpdir, "test.txt") with open(path, "wb") as file: file.write(b"") client = TestClient(app) with open(path, "rb") as f: response = client.post("/", files={"test": f}) assert response.json() == { "test": {"filename": "test.txt", "content": ""} } def test_multipart_request_multiple_files(tmpdir): path1 = os.path.join(tmpdir, "test1.txt") with open(path1, "wb") as file: file.write(b"") path2 = os.path.join(tmpdir, "test2.txt") with open(path2, "wb") as file: file.write(b"") client = TestClient(app) with open(path1, "rb") as f1, open(path2, "rb") as f2: response = client.post("/", files={"test1": f1, "test2": f2}) assert response.json() == { "test1": {"filename": "test1.txt", "content": ""}, "test2": {"filename": "test2.txt", "content": ""}, } def test_multi_items(tmpdir): path1 = os.path.join(tmpdir, "test1.txt") with open(path1, "wb") as file: file.write(b"") path2 = os.path.join(tmpdir, "test2.txt") with open(path2, "wb") as file: file.write(b"") client = TestClient(multi_items_app) with open(path1, "rb") as f1, open(path2, "rb") as f2: response = client.post( "/", data=[("test1", "abc")], files=[("test1", f1), ("test1", f2)] ) assert response.json() == { "test1": [ "abc", {"filename": "test1.txt", "content": ""}, {"filename": "test2.txt", "content": ""}, ] } def test_multipart_request_mixed_files_and_data(tmpdir): client = TestClient(app) response = client.post( "/", data=( # data b"--a7f7ac8d4e2e437c877bb7b8d7cc549c\r\n" b'Content-Disposition: form-data; name="field0"\r\n\r\n' b"value0\r\n" # file b"--a7f7ac8d4e2e437c877bb7b8d7cc549c\r\n" b'Content-Disposition: form-data; name="file"; filename="file.txt"\r\n' b"Content-Type: text/plain\r\n\r\n" b"\r\n" # data b"--a7f7ac8d4e2e437c877bb7b8d7cc549c\r\n" b'Content-Disposition: form-data; name="field1"\r\n\r\n' b"value1\r\n" b"--a7f7ac8d4e2e437c877bb7b8d7cc549c--\r\n" ), headers={ "Content-Type": "multipart/form-data; boundary=a7f7ac8d4e2e437c877bb7b8d7cc549c" }, ) assert response.json() == { "file": {"filename": "file.txt", "content": ""}, "field0": "value0", "field1": "value1", } def test_urlencoded_request_data(tmpdir): client = TestClient(app) response = client.post("/", data={"some": "data"}) assert response.json() == {"some": "data"} def test_no_request_data(tmpdir): client = TestClient(app) response = client.post("/") assert response.json() == {} def test_urlencoded_percent_encoding(tmpdir): client = TestClient(app) response = client.post("/", data={"some": "da ta"}) assert response.json() == {"some": "da ta"} def test_urlencoded_percent_encoding_keys(tmpdir): client = TestClient(app) response = client.post("/", data={"so me": "data"}) assert response.json() == {"so me": "data"} def test_urlencoded_multi_field_app_reads_body(tmpdir): client = TestClient(app_read_body) response = client.post("/", data={"some": "data", "second": "key pair"}) assert response.json() == {"some": "data", "second": "key pair"} def test_multipart_multi_field_app_reads_body(tmpdir): client = TestClient(app_read_body) response = client.post( "/", data={"some": "data", "second": "key pair"}, files=FORCE_MULTIPART ) assert response.json() == {"some": "data", "second": "key pair"}