forked from vikynandha-zz/google-drive-backup
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathdrive.py
executable file
·241 lines (190 loc) · 8.17 KB
/
drive.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
#!/usr/bin/python
"""Simple command-line sample for the Google Drive API.
Command-line application that retrieves the list of files in google drive.
Usage:
$ python drive.py
You can also get help on all the command-line flags the program understands
by running:
$ python drive.py --help
To get detailed log output run:
$ python drive.py --logging_level=DEBUG
"""
__author__ = '[email protected] (Vignesh Nandha Kumar)'
import gflags, httplib2, logging, os, pprint, sys, re, time
import pprint
import traceback
from apiclient.discovery import build
from oauth2client.file import Storage
from oauth2client.client import AccessTokenRefreshError, flow_from_clientsecrets
from oauth2client import tools
FLAGS = gflags.FLAGS
# CLIENT_SECRETS, name of a file containing the OAuth 2.0 information for this
# application, including client_id and client_secret, which are found
# on the API Access tab on the Google APIs
# Console <http://code.google.com/apis/console>
CLIENT_SECRETS = 'client_secrets.json'
# Helpful message to display in the browser if the CLIENT_SECRETS file
# is missing.
MISSING_CLIENT_SECRETS_MESSAGE = """
WARNING: Please configure OAuth 2.0
To make this sample run you will need to populate the client_secrets.json file
found at:
%s
with information from the APIs Console <https://code.google.com/apis/console>.
""" % os.path.join(os.path.dirname(__file__), CLIENT_SECRETS)
# Set up a Flow object to be used if we need to authenticate.
FLOW = flow_from_clientsecrets(CLIENT_SECRETS,
scope='https://www.googleapis.com/auth/drive',
message=MISSING_CLIENT_SECRETS_MESSAGE)
# The gflags module makes defining command-line options easy for
# applications. Run this program with the '--help' argument to see
# all the flags that it understands.
gflags.DEFINE_enum('logging_level', 'ERROR',
['DEBUG', 'INFO', 'WARNING', 'ERROR', 'CRITICAL'],
'Set the level of logging detail.')
gflags.DEFINE_string('destination', 'downloaded/', 'Destination folder location', short_name='d')
gflags.DEFINE_boolean('debug', False, 'Log folder contents as being fetched' )
gflags.DEFINE_string('logfile', 'drive.log', 'Location of file to write the log' )
gflags.DEFINE_string('drive_id', 'root', 'ID of the folder whose contents are to be fetched' )
gflags.DEFINE_boolean('noauth_local_webserver', False, 'Google authentication if the browser is not on the local system' )
def open_logfile():
if not re.match( '^/', FLAGS.logfile ):
FLAGS.logfile = FLAGS.destination + FLAGS.logfile
global LOG_FILE
LOG_FILE = open( FLAGS.logfile, 'w+b' )
def log(log):
LOG_FILE.write( (log + '\n').encode('utf8') )
def append_rename(path, count):
without_slash = path.rstrip(os.sep)
try_path = without_slash + str(count)
if os.path.exists(try_path):
append_rename(path, count + 1)
else:
print([without_slash, try_path])
os.rename(without_slash, try_path)
def mkdir_p(path):
try:
os.makedirs(path)
except OSError as exc:
if (os.path.isdir(path)):
pass
else:
# This deals with files and directories of same name in same place
append_rename(path, 0)
mkdir_p(path)
def ensure_dir(directory):
if not os.path.exists(directory):
log( "Creating directory: %s" % directory )
mkdir_p (directory)
def is_google_doc(drive_file):
return True if re.match( '^application/vnd\.google-apps\..+', drive_file['mimeType'] ) else False
def is_file_modified(drive_file, local_file):
if os.path.exists( local_file ):
rtime = time.mktime( time.strptime( drive_file['modifiedDate'], '%Y-%m-%dT%H:%M:%S.%fZ' ) )
ltime = os.path.getmtime( local_file )
return rtime > ltime
else:
return True
def get_folder_contents( service, http, folder, base_path='./', depth=0 ):
if FLAGS.debug:
log( "\n" + ' ' * depth + "Getting contents of folder %s" % folder['title'] )
try:
folder_contents = service.files().list( q="'%s' in parents" % folder['id'] ).execute()
except:
log( "ERROR: Couldn't get contents of folder %s. Retrying..." % folder['title'] )
get_folder_contents( service, http, folder, base_path, depth )
return
folder_contents = folder_contents['items']
dest_path = base_path + folder['title'].replace( '/', '_' ) + '/'
def is_file(item):
return item['mimeType'] != 'application/vnd.google-apps.folder'
def is_folder(item):
return item['mimeType'] == 'application/vnd.google-apps.folder'
if FLAGS.debug:
for item in folder_contents:
if is_folder( item ):
log( ' ' * depth + "[] " + item['title'] )
else:
log( ' ' * depth + "-- " + item['title'] )
ensure_dir( dest_path )
for item in filter(is_file, folder_contents):
full_path = dest_path + item['title'].replace( '/', '_' )
if is_file_modified( item, full_path ):
is_file_new = not os.path.exists( full_path )
if download_file( service, item, dest_path ):
if is_file_new:
log( "Created %s" % full_path )
else:
log( "Updated %s" % full_path )
else:
log( "ERROR while saving %s" % full_path )
for item in filter(is_folder, folder_contents):
get_folder_contents( service, http, item, dest_path, depth+1 )
def download_file( service, drive_file, dest_path ):
"""Download a file's content.
Args:
service: Drive API service instance.
drive_file: Drive File instance.
Returns:
File's content if successful, None otherwise.
"""
file_location = dest_path + drive_file['title'].replace( '/', '_' )
if is_google_doc(drive_file):
try:
download_url = drive_file['exportLinks']['application/pdf']
except KeyError:
download_url = None
else:
download_url = drive_file['downloadUrl']
if download_url:
try:
resp, content = service._http.request(download_url)
except Exception as e: #httplib2.IncompleteRead: # no longer exists
log( traceback.format_exc(e) + ' Error while reading file %s. Retrying...' % drive_file['title'].replace( '/', '_' ) )
download_file( service, drive_file, dest_path )
return False
if resp.status == 200:
try:
target = open( file_location, 'w+b' )
except:
log( "Could not open file %s for writing. Please check permissions." % file_location )
return False
target.write(content)
return True
else:
log( 'An error occurred: %s' % resp )
return False
else:
# The file doesn't have any content stored on Drive.
return False
def main(argv):
# Let the gflags module process the command-line arguments
try:
argv = FLAGS(argv)
except gflags.FlagsError as e:
print('%s\\nUsage: %s ARGS\\n%s' % (e, argv[0], FLAGS))
sys.exit(1)
# Set the logging according to the command-line flag
logging.getLogger().setLevel(getattr(logging, FLAGS.logging_level))
# If the Credentials don't exist or are invalid run through the native client
# flow. The Storage object will ensure that if successful the good
# Credentials will get written back to a file.
storage = Storage('drive.dat')
credentials = storage.get()
if credentials is None or credentials.invalid:
credentials = tools.run_flow(FLOW, storage)
# Create an httplib2.Http object to handle our HTTP requests and authorize it
# with our good Credentials.
http = httplib2.Http()
http = credentials.authorize(http)
service = build("drive", "v2", http=http)
mkdir_p("downloaded")
open_logfile()
try:
start_folder = service.files().get( fileId=FLAGS.drive_id ).execute()
get_folder_contents( service, http, start_folder, FLAGS.destination )
except AccessTokenRefreshError:
print ("The credentials have been revoked or expired, please re-run"
"the application to re-authorize")
if __name__ == '__main__':
main(sys.argv)