Skip to content
Merged
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
47 changes: 32 additions & 15 deletions pydrive2/files.py
Original file line number Diff line number Diff line change
@@ -1,3 +1,4 @@
import os
import io
import mimetypes
import json
Expand Down Expand Up @@ -261,7 +262,9 @@ def SetContentString(self, content, encoding="utf-8"):
:param content: content of the file in string.
:type content: str
"""
self.content = io.BytesIO(content.encode(encoding))
if content:
self.content = io.BytesIO(content.encode(encoding))

if self.get("mimeType") is None:
self["mimeType"] = "text/plain"

Expand All @@ -275,7 +278,9 @@ def SetContentFile(self, filename):
:param filename: name of the file to be uploaded.
:type filename: str.
"""
self.content = open(filename, "rb")
if os.path.getsize(filename):
self.content = open(filename, "rb")
Comment thread
efiop marked this conversation as resolved.

if self.get("title") is None:
self["title"] = filename
if self.get("mimeType") is None:
Expand Down Expand Up @@ -353,20 +358,32 @@ def download(fd, request):
download(fd, files.get_media(fileId=file_id))
except errors.HttpError as error:
exc = ApiRequestError(error)
if (
exc.error["code"] != 403
or exc.GetField("reason") != "fileNotDownloadable"
):
code = exc.error["code"]
reason = exc.GetField("reason")
if code == 403 and reason == "fileNotDownloadable":
mimetype = mimetype or "text/plain"
fd.seek(0) # just in case `download()` modified `fd`
try:
download(
fd,
files.export_media(
fileId=file_id, mimeType=mimetype
),
)
except errors.HttpError as error:
raise ApiRequestError(error)
elif code == 416 and reason == "requestedRangeNotSatisfiable":
# NOTE: An empty file case. Wasting one API call to make
# absolutely sure. See
# https://github.com/iterative/dvc/issues/4507
try:
self.FetchMetadata(fields="fileSize")
if int(self["fileSize"]) != 0:
raise exc
except errors.HttpError:
raise exc
else:
raise exc
mimetype = mimetype or "text/plain"
fd.seek(0) # just in case `download()` modified `fd`
try:
download(
fd,
files.export_media(fileId=file_id, mimeType=mimetype),
)
except errors.HttpError as error:
raise ApiRequestError(error)

if mimetype == "text/plain" and remove_bom:
fd.seek(0)
Expand Down
34 changes: 34 additions & 0 deletions pydrive2/test/test_file.py
Original file line number Diff line number Diff line change
Expand Up @@ -300,6 +300,40 @@ def test_11_Files_Get_Content_Buffer(self):

self.DeleteUploadedFiles(drive, [file1["id"]])

def test_12_Upload_Download_Empty_File(self):
filename = os.path.join(self.tmpdir, str(time()))
create_file(filename, "")

drive = GoogleDrive(self.ga)
file1 = drive.CreateFile()
file1.SetContentFile(filename)
pydrive_retry(file1.Upload)

fileOut1 = self.getTempFile()
pydrive_retry(file1.GetContentFile, fileOut1)
self.assertEqual(os.path.getsize(fileOut1), 0)

self.DeleteUploadedFiles(drive, [file1["id"]])

def test_13_Upload_Download_Empty_String(self):
drive = GoogleDrive(self.ga)
file1 = drive.CreateFile()
file1.SetContentString("")
pydrive_retry(file1.Upload)

self.assertEqual(pydrive_retry(file1.GetContentString), "")

# Force download and double check content
pydrive_retry(file1.FetchContent)
self.assertEqual(file1.GetContentString(), "")

# Download file from id
file2 = drive.CreateFile({"id": file1["id"]})
pydrive_retry(file2.FetchContent)
self.assertEqual(file2.GetContentString(), "")

self.DeleteUploadedFiles(drive, [file1["id"]])

# Tests for Trash/UnTrash/Delete.
# ===============================

Expand Down