forked from matthoskins1980/Flask-Tus
-
Notifications
You must be signed in to change notification settings - Fork 3
/
flask_tus.py
312 lines (242 loc) · 12.5 KB
/
flask_tus.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
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
from flask import Blueprint, request, jsonify, make_response, current_app
import base64
import os
import redis
import uuid
# Find the stack on which we want to store the database connection.
# Starting with Flask 0.9, the _app_ctx_stack is the correct one,
# before that we need to use the _request_ctx_stack.
try:
from flask import _app_ctx_stack as stack
except ImportError:
from flask import _request_ctx_stack as stack
class TusManager(object):
def __init__(self, app=None, upload_url='/file-upload', upload_folder='uploads/', overwrite=True,
upload_finish_cb=None):
self.tus_api_version = '1.0.0'
self.tus_api_version_supported = '1.0.0'
self.tus_api_extensions = ['creation', 'termination', 'file-check']
self.tus_max_file_size = 4294967296 # 4GByte
self.file_overwrite = overwrite
self.upload_finish_cb = upload_finish_cb
self.upload_file_handler_cb = None
self.blueprint = Blueprint('tus-manager', __name__)
if app is not None:
self.init_app(app, upload_url, upload_folder)
def init_app(self, app, upload_url=None, upload_folder=None):
if upload_url is not None:
self.upload_url = upload_url
if upload_folder is not None:
self.upload_folder = upload_folder
self._register_routes()
self.app = app
self.app.register_blueprint(self.blueprint)
def _register_routes(self):
# Note on routing names and method names.
# Use '-' to separate words/parts in endpoint.
# Use '_" to separate words/parts in method name.
# Use "tus" to start name
# followed by "core" or the extension name
# followed by protocol version (ommitting any trailing ".0"s and replacing '.' with '-' or '_')
# followed by a brief human redable description of what this method does
#
# Use "proprietary" instead of "core" or extension name for any features that are proprietary to this
# implementation.
# routes without resource id
self.blueprint.add_url_rule(
self.upload_url, 'tus-proprietary-get-file-exists', self.tus_proprietary_get_file_exists, methods=['GET'])
self.blueprint.add_url_rule(
self.upload_url, 'tus-1-get-server-info', self.tus_1_get_server_info, methods=['OPTIONS'])
# Tus protocol docs Creation section 6.a.iii
self.blueprint.add_url_rule(
self.upload_url, 'tus-creation-1-create', self.tus_creation_1_create, methods=['POST'])
# routes with resource id
# Tus protocol docs Core section 5.c.i.
self.blueprint.add_url_rule(
f"{self.upload_url}/<resource_id>", 'tus-1-get-resume-info', self.tus_1_get_resume_info, methods=['HEAD'])
# Tus protocol docs Core section 5.c.ii
self.blueprint.add_url_rule(
f"{self.upload_url}/<resource_id>", 'tus-1-upload-chunk', self.tus_1_upload_chunk, methods=['PATCH'])
# Tus protocol docs Termination section 6.e.i
self.blueprint.add_url_rule(
f"{self.upload_url}/<resource_id>", 'tus-1-delete', self.tus_1_delete, methods=['DELETE'])
def upload_file_handler(self, callback):
self.upload_file_handler_cb = callback
return callback
# handle redis server connection
def redis_connect(self):
return redis.Redis()
@property
def redis_connection(self):
ctx = stack.top
if ctx is not None:
if not hasattr(ctx, 'tus_redis'):
ctx.tus_redis = self.redis_connect()
return ctx.tus_redis
def _parse_metadata(self):
metadata = {}
for kv in request.headers.get("Upload-Metadata", None).split(","):
(key, value) = kv.split(" ")
metadata[key] = base64.b64decode(value).decode('utf-8')
return metadata
# Untested. Possibly unused.
def tus_proprietary_get_file_exists(self):
print('begin tus_proprietary_get_file_exists')
response = make_response("", 200)
if 'Upload-Metadata' in request.headers:
response.headers['Upload-Metadata'] = request.headers['Upload-Metadata']
metadata = self._parse_metadata()
if metadata.get("filename", None) is None:
return make_response("metadata filename is not set", 404)
(filename_name, extension) = os.path.splitext(metadata.get("filename"))
if filename_name.upper() in [os.path.splitext(f)[0].upper() for f in
os.listdir(os.path.dirname(self.upload_folder))]:
response.headers['Tus-File-Name'] = metadata.get("filename")
response.headers['Tus-File-Exists'] = True
else:
response.headers['Tus-File-Exists'] = False
return response
def create_resource_id(self):
return str(uuid.uuid4())
def create_url(self, resource_id):
return '{}/{}/{}'.format(request.url_root, self.upload_url, resource_id)
def tus_creation_1_create(self):
"""Implements POST to create file according to Tus protocol Creation extension"""
print('begin tus_creation_1_create')
response = make_response("", 200)
response.headers['Tus-Resumable'] = self.tus_api_version
response.headers['Tus-Version'] = self.tus_api_version_supported
if 'Upload-Metadata' in request.headers:
response.headers['Upload-Metadata'] = request.headers['Upload-Metadata']
if request.headers.get("Tus-Resumable") is None:
self.app.logger.warning("Received File upload for unsupported file transfer protocol")
response.data = "Received File upload for unsupported file transfer protocol"
response.status_code = 500
return response
# process upload metadata
metadata = self._parse_metadata()
if os.path.lexists(
os.path.join(self.upload_folder, metadata.get("filename"))) and self.file_overwrite is False:
response.status_code = 409
return response
file_size = int(request.headers.get("Upload-Length", "0"))
resource_id = self.create_resource_id()
p = self.redis_connection.pipeline()
p.setex("file-uploads/{}/filename".format(resource_id), 3600, "{}".format(metadata.get("filename")))
p.setex("file-uploads/{}/file_size".format(resource_id), 3600, file_size)
p.setex("file-uploads/{}/offset".format(resource_id), 3600, 0)
p.setex("file-uploads/{}/upload-metadata".format(resource_id), 3600, request.headers.get("Upload-Metadata"))
p.execute()
try:
open(os.path.join(self.upload_folder, resource_id), "w").close()
except IOError as e:
self.app.logger.error("Unable to create file: {}".format(e))
response.status_code = 500
return response
response.status_code = 201
response.headers['Location'] = self.create_url(resource_id)
response.headers['Tus-Temp-Filename'] = resource_id
response.autocorrect_location_header = False
return response
# Untested. should be part of protocol, identify sections and verify correctness
def tus_1_get_server_info(self):
print('begin tus_1_get_server_info')
response = make_response("", 200)
if 'Upload-Metadata' in request.headers:
response.headers['Upload-Metadata'] = request.headers['Upload-Metadata']
if request.headers.get('Access-Control-Request-Method', None) is not None:
# CORS option request, return 200
return response
if request.headers.get("Tus-Resumable") is not None:
response.headers['Tus-Resumable'] = self.tus_api_version
response.headers['Tus-Version'] = self.tus_api_version_supported
response.headers['Tus-Extension'] = ",".join(self.tus_api_extensions)
response.headers['Tus-Max-Size'] = self.tus_max_file_size
response.status_code = 204
return response
else:
self.app.logger.warning("Received File upload for unsupported file transfer protocol")
response.data = "Received File upload for unsupported file transfer protocol"
response.status_code = 500
return response
def tus_1_get_resume_info(self, resource_id):
"""Implements HEAD according to Tus Core protocol."""
response = make_response("", 204)
response.headers['Tus-Resumable'] = self.tus_api_version
response.headers['Tus-Version'] = self.tus_api_version_supported
response.headers['Cache-Control'] = 'no-store'
offset = self.redis_connection.get("file-uploads/{}/offset".format(resource_id))
length = self.redis_connection.get("file-uploads/{}/file_size".format(resource_id))
metadata = self.redis_connection.get("file-uploads/{}/upload-metadata".format(resource_id))
if offset is None:
response.status_code = 404
return response
response.status_code = 200
response.headers['Upload-Offset'] = offset
if length is not None:
response.headers['Upload-Length'] = length
if metadata is not None:
response.headers['Upload-Metadata'] = request.headers['Upload-Metadata']
return response
def tus_1_upload_chunk(self, resource_id):
"""Implements DELETE according to Tus Core protocol"""
response = make_response("", 204)
response.headers['Tus-Resumable'] = self.tus_api_version
response.headers['Tus-Version'] = self.tus_api_version_supported
# TODO: update following variable names to reflect "ours" (from redis) and "supplied" from headers
filename = self.redis_connection.get("file-uploads/{}/filename".format(resource_id)).decode('utf-8')
file_size = int(self.redis_connection.get("file-uploads/{}/file_size".format(resource_id)))
redis_offset = self.redis_connection.get("file-uploads/{}/offset".format(resource_id)).decode('utf-8')
file_offset = int(request.headers.get("Upload-Offset", 0))
chunk_size = int(request.headers.get("Content-Length", 0))
header_offset = request.headers.get("Upload-Offset")
upload_file_path = os.path.join(self.upload_folder, resource_id)
if filename is None or os.path.lexists(upload_file_path) is False:
self.app.logger.info("PATCH sent for resource_id that does not exist. {}".format(resource_id))
response.status_code = 410
return response
if header_offset != redis_offset:
response.status_code = 409 # HTTP 409 Conflict
return response
try:
f = open(upload_file_path, "r+b")
except IOError:
f = open(upload_file_path, "wb")
finally:
f.seek(file_offset)
f.write(request.data)
f.close()
new_offset = self.redis_connection.incrby("file-uploads/{}/offset".format(resource_id), chunk_size)
response.headers['Upload-Offset'] = new_offset
response.headers['Tus-Temp-Filename'] = resource_id
if file_size == new_offset: # file transfer complete, rename from resource id to actual filename
try:
if self.upload_file_handler_cb is None:
os.rename(upload_file_path, os.path.join(self.upload_folder, filename))
else:
filename = self.upload_file_handler_cb(upload_file_path, filename)
finally:
self._remove_resources(resource_id)
if self.upload_finish_cb is not None:
self.upload_finish_cb()
return response
def _remove_resources(self, resource_id):
p = self.redis_connection.pipeline()
p.delete("file-uploads/{}/filename".format(resource_id))
p.delete("file-uploads/{}/file_size".format(resource_id))
p.delete("file-uploads/{}/offset".format(resource_id))
p.delete("file-uploads/{}/upload-metadata".format(resource_id))
p.execute()
upload_file_path = os.path.join(self.upload_folder, resource_id)
try:
os.remove(upload_file_path)
except FileNotFoundError:
pass
def tus_1_delete(self, resource_id):
"""Implements DELETE according to Tus Termination protocol"""
response = make_response("", 204)
response.headers['Tus-Resumable'] = self.tus_api_version
response.headers['Tus-Version'] = self.tus_api_version_supported
self._remove_resources(resource_id)
response.status_code = 204
return respose