2019-08-28 22:37:56 +02:00
|
|
|
from __future__ import unicode_literals
|
2014-07-30 22:16:58 +02:00
|
|
|
import zipfile
|
2017-12-29 20:22:49 +01:00
|
|
|
import os.path
|
2017-01-31 15:08:00 +01:00
|
|
|
import sys
|
2019-11-22 21:53:20 +01:00
|
|
|
import xbmc, xbmcvfs, xbmcgui
|
|
|
|
from . import dropbox
|
2019-08-27 21:55:22 +02:00
|
|
|
from . import utils as utils
|
2019-11-22 21:53:20 +01:00
|
|
|
from .dropbox.files import WriteMode,CommitInfo,UploadSessionCursor
|
2019-09-05 21:30:06 +02:00
|
|
|
from . authorizers import DropboxAuthorizer
|
2012-10-26 16:42:10 +02:00
|
|
|
|
|
|
|
class Vfs:
|
2012-11-04 21:39:17 +01:00
|
|
|
root_path = None
|
2012-10-26 16:42:10 +02:00
|
|
|
|
2013-01-28 22:44:22 +01:00
|
|
|
def __init__(self,rootString):
|
|
|
|
self.set_root(rootString)
|
|
|
|
|
2012-11-04 21:39:17 +01:00
|
|
|
def set_root(self,rootString):
|
2012-11-05 18:13:48 +01:00
|
|
|
old_root = self.root_path
|
2012-11-04 21:39:17 +01:00
|
|
|
self.root_path = rootString
|
2014-08-01 00:29:59 +02:00
|
|
|
|
2019-11-25 22:19:57 +01:00
|
|
|
# fix slashes
|
2012-11-04 21:39:17 +01:00
|
|
|
self.root_path = self.root_path.replace("\\","/")
|
|
|
|
|
2019-11-25 22:19:57 +01:00
|
|
|
# check if trailing slash is included
|
2012-11-04 21:39:17 +01:00
|
|
|
if(self.root_path[-1:] != "/"):
|
|
|
|
self.root_path = self.root_path + "/"
|
2012-11-05 18:13:48 +01:00
|
|
|
|
2019-11-25 22:19:57 +01:00
|
|
|
# return the old root
|
2012-11-05 18:13:48 +01:00
|
|
|
return old_root
|
2012-11-04 21:39:17 +01:00
|
|
|
|
|
|
|
def listdir(self,directory):
|
2012-10-26 16:42:10 +02:00
|
|
|
return {}
|
|
|
|
|
2012-11-04 21:39:17 +01:00
|
|
|
def mkdir(self,directory):
|
2012-10-26 16:42:10 +02:00
|
|
|
return True
|
|
|
|
|
2012-11-06 18:37:39 +01:00
|
|
|
def put(self,source,dest):
|
2012-10-26 16:42:10 +02:00
|
|
|
return True
|
2012-11-02 21:59:40 +01:00
|
|
|
|
2012-11-04 21:39:17 +01:00
|
|
|
def rmdir(self,directory):
|
2012-11-02 21:59:40 +01:00
|
|
|
return True
|
|
|
|
|
2014-08-01 20:12:43 +02:00
|
|
|
def rmfile(self,aFile):
|
|
|
|
return True
|
|
|
|
|
2012-11-04 21:39:17 +01:00
|
|
|
def exists(self,aFile):
|
2012-11-02 21:59:40 +01:00
|
|
|
return True
|
2014-07-30 22:16:58 +02:00
|
|
|
|
|
|
|
def rename(self,aFile,newName):
|
|
|
|
return True
|
|
|
|
|
|
|
|
def cleanup(self):
|
|
|
|
return True
|
2012-10-26 16:42:10 +02:00
|
|
|
|
2012-11-02 21:59:40 +01:00
|
|
|
class XBMCFileSystem(Vfs):
|
2013-01-28 22:44:22 +01:00
|
|
|
|
2012-11-04 21:39:17 +01:00
|
|
|
def listdir(self,directory):
|
2012-11-02 21:59:40 +01:00
|
|
|
return xbmcvfs.listdir(directory)
|
|
|
|
|
2012-11-04 21:39:17 +01:00
|
|
|
def mkdir(self,directory):
|
2014-08-08 17:37:16 +02:00
|
|
|
return xbmcvfs.mkdir(xbmc.translatePath(directory))
|
2012-11-02 21:59:40 +01:00
|
|
|
|
2012-11-06 18:37:39 +01:00
|
|
|
def put(self,source,dest):
|
2014-08-08 17:37:16 +02:00
|
|
|
return xbmcvfs.copy(xbmc.translatePath(source),xbmc.translatePath(dest))
|
2012-11-06 18:37:39 +01:00
|
|
|
|
2012-11-04 21:39:17 +01:00
|
|
|
def rmdir(self,directory):
|
2012-11-02 21:59:40 +01:00
|
|
|
return xbmcvfs.rmdir(directory,True)
|
|
|
|
|
2014-08-01 20:12:43 +02:00
|
|
|
def rmfile(self,aFile):
|
|
|
|
return xbmcvfs.delete(aFile)
|
|
|
|
|
2014-07-30 22:16:58 +02:00
|
|
|
def rename(self,aFile,newName):
|
|
|
|
return xbmcvfs.rename(aFile, newName)
|
|
|
|
|
2012-11-04 21:39:17 +01:00
|
|
|
def exists(self,aFile):
|
2012-11-02 21:59:40 +01:00
|
|
|
return xbmcvfs.exists(aFile)
|
|
|
|
|
2014-07-30 22:16:58 +02:00
|
|
|
class ZipFileSystem(Vfs):
|
|
|
|
zip = None
|
|
|
|
|
|
|
|
def __init__(self,rootString,mode):
|
|
|
|
self.root_path = ""
|
2015-04-24 16:23:56 +02:00
|
|
|
self.zip = zipfile.ZipFile(rootString,mode=mode,compression=zipfile.ZIP_DEFLATED,allowZip64=True)
|
2014-07-30 22:16:58 +02:00
|
|
|
|
|
|
|
def listdir(self,directory):
|
2014-08-01 19:57:55 +02:00
|
|
|
return [[],[]]
|
2014-07-30 22:16:58 +02:00
|
|
|
|
|
|
|
def mkdir(self,directory):
|
2019-11-25 22:19:57 +01:00
|
|
|
# self.zip.write(directory[len(self.root_path):])
|
2014-08-01 19:57:55 +02:00
|
|
|
return False
|
2014-07-30 22:16:58 +02:00
|
|
|
|
|
|
|
def put(self,source,dest):
|
2014-08-08 17:26:39 +02:00
|
|
|
|
|
|
|
aFile = xbmcvfs.File(xbmc.translatePath(source),'r')
|
|
|
|
|
2019-08-28 22:37:56 +02:00
|
|
|
self.zip.writestr(dest,aFile.readBytes())
|
2014-08-08 17:26:39 +02:00
|
|
|
|
2014-07-30 22:16:58 +02:00
|
|
|
return True
|
|
|
|
|
|
|
|
def rmdir(self,directory):
|
2014-08-01 19:57:55 +02:00
|
|
|
return False
|
2014-07-30 22:16:58 +02:00
|
|
|
|
|
|
|
def exists(self,aFile):
|
2014-08-01 19:57:55 +02:00
|
|
|
return False
|
2014-07-30 22:16:58 +02:00
|
|
|
|
|
|
|
def cleanup(self):
|
|
|
|
self.zip.close()
|
2014-08-01 19:57:55 +02:00
|
|
|
|
2015-06-26 22:32:22 +02:00
|
|
|
def extract(self,aFile,path):
|
2019-11-25 22:19:57 +01:00
|
|
|
# extract zip file to path
|
2015-06-26 22:32:22 +02:00
|
|
|
self.zip.extract(aFile,path)
|
|
|
|
|
|
|
|
def listFiles(self):
|
|
|
|
return self.zip.infolist()
|
2014-07-30 22:16:58 +02:00
|
|
|
|
2012-11-04 21:39:17 +01:00
|
|
|
class DropboxFileSystem(Vfs):
|
2019-11-25 22:19:57 +01:00
|
|
|
MAX_CHUNK = 50 * 1000 * 1000 # dropbox uses 150, reduced to 50 for small mem systems
|
2012-11-05 19:25:25 +01:00
|
|
|
client = None
|
2014-10-09 21:30:39 +02:00
|
|
|
APP_KEY = ''
|
|
|
|
APP_SECRET = ''
|
2012-11-04 21:39:17 +01:00
|
|
|
|
2013-01-28 22:44:22 +01:00
|
|
|
def __init__(self,rootString):
|
2013-03-04 17:22:09 +01:00
|
|
|
self.set_root(rootString)
|
2013-01-28 22:44:22 +01:00
|
|
|
|
2017-01-31 15:08:00 +01:00
|
|
|
authorizer = DropboxAuthorizer()
|
2012-11-05 19:25:25 +01:00
|
|
|
|
2017-01-31 15:08:00 +01:00
|
|
|
if(authorizer.isAuthorized()):
|
|
|
|
self.client = authorizer.getClient()
|
2012-11-05 19:25:25 +01:00
|
|
|
else:
|
2019-11-25 22:19:57 +01:00
|
|
|
# tell the user to go back and run the authorizer
|
2017-01-31 15:16:54 +01:00
|
|
|
xbmcgui.Dialog().ok(utils.getString(30010),utils.getString(30105))
|
2017-01-31 15:08:00 +01:00
|
|
|
sys.exit()
|
2012-11-05 19:25:25 +01:00
|
|
|
|
2012-11-05 22:46:17 +01:00
|
|
|
def listdir(self,directory):
|
2017-12-04 00:32:21 +01:00
|
|
|
directory = self._fix_slashes(directory)
|
|
|
|
|
2012-11-05 22:46:17 +01:00
|
|
|
if(self.client != None and self.exists(directory)):
|
|
|
|
files = []
|
|
|
|
dirs = []
|
2017-12-04 00:32:21 +01:00
|
|
|
metadata = self.client.files_list_folder(directory)
|
2012-11-05 22:46:17 +01:00
|
|
|
|
2017-12-04 00:32:21 +01:00
|
|
|
for aFile in metadata.entries:
|
|
|
|
if(isinstance(aFile,dropbox.files.FolderMetadata)):
|
2019-08-28 22:37:56 +02:00
|
|
|
dirs.append(aFile.name)
|
2012-11-05 22:46:17 +01:00
|
|
|
else:
|
2019-08-28 22:37:56 +02:00
|
|
|
files.append(aFile.name)
|
2012-11-05 22:46:17 +01:00
|
|
|
|
|
|
|
return [dirs,files]
|
|
|
|
else:
|
|
|
|
return [[],[]]
|
|
|
|
|
|
|
|
|
2012-11-05 19:25:25 +01:00
|
|
|
def mkdir(self,directory):
|
2013-03-04 17:22:09 +01:00
|
|
|
directory = self._fix_slashes(directory)
|
2012-11-05 19:25:25 +01:00
|
|
|
if(self.client != None):
|
2019-11-25 22:19:57 +01:00
|
|
|
# sort of odd but always return true, folder create is implicit with file upload
|
2012-11-05 19:25:25 +01:00
|
|
|
return True
|
|
|
|
else:
|
|
|
|
return False
|
|
|
|
|
2012-11-05 22:46:17 +01:00
|
|
|
def rmdir(self,directory):
|
2013-03-04 17:22:09 +01:00
|
|
|
directory = self._fix_slashes(directory)
|
2012-11-05 22:46:17 +01:00
|
|
|
if(self.client != None and self.exists(directory)):
|
2019-11-25 22:19:57 +01:00
|
|
|
# dropbox is stupid and will refuse to do this sometimes, need to delete recursively
|
2014-07-29 15:43:34 +02:00
|
|
|
dirs,files = self.listdir(directory)
|
|
|
|
|
|
|
|
for aDir in dirs:
|
|
|
|
self.rmdir(aDir)
|
|
|
|
|
2019-11-25 22:19:57 +01:00
|
|
|
# finally remove the root directory
|
2017-12-04 00:32:21 +01:00
|
|
|
self.client.files_delete(directory)
|
2014-08-01 20:12:43 +02:00
|
|
|
|
|
|
|
return True
|
|
|
|
else:
|
|
|
|
return False
|
|
|
|
|
2014-11-21 20:30:22 +01:00
|
|
|
def rmfile(self,aFile):
|
2014-08-01 20:12:43 +02:00
|
|
|
aFile = self._fix_slashes(aFile)
|
2014-11-21 20:30:22 +01:00
|
|
|
|
2014-08-01 20:12:43 +02:00
|
|
|
if(self.client != None and self.exists(aFile)):
|
2017-12-04 00:32:21 +01:00
|
|
|
self.client.files_delete(aFile)
|
2014-08-01 20:12:43 +02:00
|
|
|
return True
|
2012-11-05 22:46:17 +01:00
|
|
|
else:
|
|
|
|
return False
|
|
|
|
|
2012-11-05 19:25:25 +01:00
|
|
|
def exists(self,aFile):
|
2013-03-04 17:22:09 +01:00
|
|
|
aFile = self._fix_slashes(aFile)
|
2017-12-27 17:06:07 +01:00
|
|
|
|
2012-11-05 19:25:25 +01:00
|
|
|
if(self.client != None):
|
2019-11-25 22:19:57 +01:00
|
|
|
# can't list root metadata
|
2017-12-04 00:32:21 +01:00
|
|
|
if(aFile == ''):
|
|
|
|
return True
|
|
|
|
|
2012-11-04 21:39:17 +01:00
|
|
|
try:
|
2017-12-04 00:32:21 +01:00
|
|
|
meta_data = self.client.files_get_metadata(aFile)
|
2019-11-25 22:19:57 +01:00
|
|
|
# if we make it here the file does exist
|
2012-11-05 19:25:25 +01:00
|
|
|
return True
|
2012-11-04 21:39:17 +01:00
|
|
|
except:
|
2012-11-05 19:25:25 +01:00
|
|
|
return False
|
|
|
|
else:
|
|
|
|
return False
|
2012-11-04 21:39:17 +01:00
|
|
|
|
2014-06-03 20:34:02 +02:00
|
|
|
def put(self,source,dest,retry=True):
|
2013-03-04 17:22:09 +01:00
|
|
|
dest = self._fix_slashes(dest)
|
|
|
|
|
2012-11-05 19:25:25 +01:00
|
|
|
if(self.client != None):
|
2019-11-25 22:19:57 +01:00
|
|
|
# open the file and get its size
|
2012-11-05 19:25:25 +01:00
|
|
|
f = open(source,'rb')
|
2017-12-29 20:22:49 +01:00
|
|
|
f_size = os.path.getsize(source)
|
|
|
|
|
2012-11-05 22:46:17 +01:00
|
|
|
try:
|
2017-12-29 20:22:49 +01:00
|
|
|
if(f_size < self.MAX_CHUNK):
|
2019-11-25 22:19:57 +01:00
|
|
|
# use the regular upload
|
2017-12-29 20:22:49 +01:00
|
|
|
response = self.client.files_upload(f.read(),dest,mode=WriteMode('overwrite'))
|
|
|
|
else:
|
2019-11-25 22:19:57 +01:00
|
|
|
# start the upload session
|
2017-12-29 20:22:49 +01:00
|
|
|
upload_session = self.client.files_upload_session_start(f.read(self.MAX_CHUNK))
|
|
|
|
upload_cursor = UploadSessionCursor(upload_session.session_id,f.tell())
|
|
|
|
|
|
|
|
while(f.tell() < f_size):
|
2019-11-25 22:19:57 +01:00
|
|
|
# check if we should finish the upload
|
2017-12-29 20:22:49 +01:00
|
|
|
if((f_size - f.tell()) <= self.MAX_CHUNK):
|
2019-11-25 22:19:57 +01:00
|
|
|
# upload and close
|
2017-12-29 20:22:49 +01:00
|
|
|
self.client.files_upload_session_finish(f.read(self.MAX_CHUNK),upload_cursor,CommitInfo(dest,mode=WriteMode('overwrite')))
|
|
|
|
else:
|
2019-11-25 22:19:57 +01:00
|
|
|
# upload a part and store the offset
|
2017-12-29 20:22:49 +01:00
|
|
|
self.client.files_upload_session_append_v2(f.read(self.MAX_CHUNK),upload_cursor)
|
|
|
|
upload_cursor.offset = f.tell()
|
|
|
|
|
2019-11-25 22:19:57 +01:00
|
|
|
# if no errors we're good!
|
2012-11-05 22:46:17 +01:00
|
|
|
return True
|
2017-12-29 20:22:49 +01:00
|
|
|
except Exception as anError:
|
|
|
|
utils.log(str(anError))
|
|
|
|
|
2019-11-25 22:19:57 +01:00
|
|
|
# if we have an exception retry
|
2014-06-03 20:34:02 +02:00
|
|
|
if(retry):
|
|
|
|
return self.put(source,dest,False)
|
|
|
|
else:
|
2019-11-25 22:19:57 +01:00
|
|
|
# tried once already, just quit
|
2014-06-03 20:34:02 +02:00
|
|
|
return False
|
2012-11-05 19:25:25 +01:00
|
|
|
else:
|
|
|
|
return False
|
2012-11-06 18:37:39 +01:00
|
|
|
|
|
|
|
def get_file(self,source,dest):
|
|
|
|
if(self.client != None):
|
2019-11-25 22:19:57 +01:00
|
|
|
# write the file locally
|
2017-12-04 00:32:21 +01:00
|
|
|
f = self.client.files_download_to_file(dest,source)
|
2013-03-04 17:27:25 +01:00
|
|
|
return True
|
2012-11-06 18:37:39 +01:00
|
|
|
else:
|
|
|
|
return False
|
2013-03-04 17:22:09 +01:00
|
|
|
|
|
|
|
def _fix_slashes(self,filename):
|
2017-12-04 00:32:21 +01:00
|
|
|
result = filename.replace('\\','/')
|
|
|
|
|
2019-11-25 22:19:57 +01:00
|
|
|
# root needs to be a blank string
|
2017-12-04 00:32:21 +01:00
|
|
|
if(result == '/'):
|
|
|
|
result = ""
|
|
|
|
|
2019-11-25 22:19:57 +01:00
|
|
|
# if dir ends in slash, remove it
|
2017-12-08 15:46:12 +01:00
|
|
|
if(result[-1:] == "/"):
|
|
|
|
result = result[:-1]
|
|
|
|
|
2017-12-04 00:32:21 +01:00
|
|
|
return result
|