2008-05-19 19:57:04 +00:00
|
|
|
|
2017-01-19 22:39:53 +00:00
|
|
|
import json
|
2008-05-19 19:57:04 +00:00
|
|
|
|
2008-10-28 20:41:04 +00:00
|
|
|
from twisted.web import http, static
|
2008-05-19 19:57:04 +00:00
|
|
|
from twisted.internet import defer
|
2019-09-10 23:52:20 +00:00
|
|
|
from twisted.web.resource import Resource
|
|
|
|
|
|
|
|
from nevow import url
|
2008-05-19 19:57:04 +00:00
|
|
|
|
2014-10-07 19:11:31 +00:00
|
|
|
from allmydata.interfaces import ExistingChildError
|
2008-10-22 08:38:18 +00:00
|
|
|
from allmydata.monitor import Monitor
|
2008-07-16 20:14:39 +00:00
|
|
|
from allmydata.immutable.upload import FileHandle
|
2011-08-07 00:43:48 +00:00
|
|
|
from allmydata.mutable.publish import MutableFileHandle
|
|
|
|
from allmydata.mutable.common import MODE_READ
|
2008-10-29 01:01:03 +00:00
|
|
|
from allmydata.util import log, base32
|
2011-08-24 15:59:28 +00:00
|
|
|
from allmydata.util.encodingutil import quote_output
|
|
|
|
from allmydata.blacklist import FileProhibited, ProhibitedNode
|
2008-05-19 19:57:04 +00:00
|
|
|
|
2009-02-20 19:15:54 +00:00
|
|
|
from allmydata.web.common import text_plain, WebError, RenderMixin, \
|
2009-03-04 04:56:30 +00:00
|
|
|
boolean_of_arg, get_arg, should_create_intermediate_directories, \
|
2011-08-07 00:43:48 +00:00
|
|
|
MyExceptionHandler, parse_replace_arg, parse_offset_arg, \
|
2014-10-07 19:11:31 +00:00
|
|
|
get_format, get_mutable_type, get_filenode_metadata
|
2012-04-03 03:02:59 +00:00
|
|
|
from allmydata.web.check_results import CheckResultsRenderer, \
|
|
|
|
CheckAndRepairResultsRenderer, LiteralCheckResultsRenderer
|
2008-09-18 05:00:41 +00:00
|
|
|
from allmydata.web.info import MoreInfo
|
2008-05-19 19:57:04 +00:00
|
|
|
|
2019-05-15 06:17:44 +00:00
|
|
|
class ReplaceMeMixin(object):
|
2009-02-20 19:15:54 +00:00
|
|
|
def replace_me_with_a_child(self, req, client, replace):
|
2008-05-19 19:57:04 +00:00
|
|
|
# a new file is being uploaded in our place.
|
2011-10-13 16:29:51 +00:00
|
|
|
file_format = get_format(req, "CHK")
|
2011-10-13 16:32:29 +00:00
|
|
|
mutable_type = get_mutable_type(file_format)
|
|
|
|
if mutable_type is not None:
|
2011-08-07 00:43:48 +00:00
|
|
|
data = MutableFileHandle(req.content)
|
|
|
|
d = client.create_mutable_file(data, version=mutable_type)
|
2008-05-20 19:36:02 +00:00
|
|
|
def _uploaded(newnode):
|
|
|
|
d2 = self.parentnode.set_node(self.name, newnode,
|
|
|
|
overwrite=replace)
|
|
|
|
d2.addCallback(lambda res: newnode)
|
|
|
|
return d2
|
|
|
|
d.addCallback(_uploaded)
|
|
|
|
else:
|
2011-10-13 16:29:51 +00:00
|
|
|
assert file_format == "CHK"
|
2008-05-20 19:36:02 +00:00
|
|
|
uploadable = FileHandle(req.content, convergence=client.convergence)
|
|
|
|
d = self.parentnode.add_file(self.name, uploadable,
|
|
|
|
overwrite=replace)
|
2008-05-19 19:57:04 +00:00
|
|
|
def _done(filenode):
|
|
|
|
log.msg("webish upload complete",
|
2010-08-06 07:07:05 +00:00
|
|
|
facility="tahoe.webish", level=log.NOISY, umid="TCjBGQ")
|
2008-05-19 19:57:04 +00:00
|
|
|
if self.node:
|
|
|
|
# we've replaced an existing file (or modified a mutable
|
|
|
|
# file), so the response code is 200
|
|
|
|
req.setResponseCode(http.OK)
|
|
|
|
else:
|
|
|
|
# we've created a new file, so the code is 201
|
|
|
|
req.setResponseCode(http.CREATED)
|
|
|
|
return filenode.get_uri()
|
|
|
|
d.addCallback(_done)
|
|
|
|
return d
|
|
|
|
|
2009-02-20 19:15:54 +00:00
|
|
|
def replace_me_with_a_childcap(self, req, client, replace):
|
2008-05-19 19:57:04 +00:00
|
|
|
req.content.seek(0)
|
|
|
|
childcap = req.content.read()
|
2010-01-27 06:44:30 +00:00
|
|
|
childnode = client.create_node_from_uri(childcap, None, name=self.name)
|
2008-05-19 19:57:04 +00:00
|
|
|
d = self.parentnode.set_node(self.name, childnode, overwrite=replace)
|
|
|
|
d.addCallback(lambda res: childnode.get_uri())
|
|
|
|
return d
|
|
|
|
|
|
|
|
|
2009-02-20 19:15:54 +00:00
|
|
|
def replace_me_with_a_formpost(self, req, client, replace):
|
2008-05-19 19:57:04 +00:00
|
|
|
# create a new file, maybe mutable, maybe immutable
|
2011-10-13 16:29:51 +00:00
|
|
|
file_format = get_format(req, "CHK")
|
2011-08-07 00:43:48 +00:00
|
|
|
contents = req.fields["file"]
|
2011-10-13 16:29:51 +00:00
|
|
|
if file_format in ("SDMF", "MDMF"):
|
|
|
|
mutable_type = get_mutable_type(file_format)
|
2011-08-07 00:43:48 +00:00
|
|
|
uploadable = MutableFileHandle(contents.file)
|
|
|
|
d = client.create_mutable_file(uploadable, version=mutable_type)
|
2008-05-19 19:57:04 +00:00
|
|
|
def _uploaded(newnode):
|
|
|
|
d2 = self.parentnode.set_node(self.name, newnode,
|
|
|
|
overwrite=replace)
|
|
|
|
d2.addCallback(lambda res: newnode.get_uri())
|
|
|
|
return d2
|
|
|
|
d.addCallback(_uploaded)
|
|
|
|
return d
|
2011-08-07 00:43:48 +00:00
|
|
|
|
2008-05-19 19:57:04 +00:00
|
|
|
uploadable = FileHandle(contents.file, convergence=client.convergence)
|
|
|
|
d = self.parentnode.add_file(self.name, uploadable, overwrite=replace)
|
|
|
|
d.addCallback(lambda newnode: newnode.get_uri())
|
|
|
|
return d
|
|
|
|
|
2011-08-07 00:43:48 +00:00
|
|
|
|
2019-09-10 23:52:20 +00:00
|
|
|
# XXX
|
|
|
|
class PlaceHolderNodeHandler(Resource, ReplaceMeMixin, object):
|
2009-02-20 19:15:54 +00:00
|
|
|
def __init__(self, client, parentnode, name):
|
2019-09-10 23:52:20 +00:00
|
|
|
super(PlaceHolderNodeHandler, self).__init__()
|
2009-02-20 19:15:54 +00:00
|
|
|
self.client = client
|
2008-05-19 19:57:04 +00:00
|
|
|
assert parentnode
|
|
|
|
self.parentnode = parentnode
|
|
|
|
self.name = name
|
|
|
|
self.node = None
|
|
|
|
|
2019-09-10 23:52:20 +00:00
|
|
|
def render_PUT(self, req):
|
2008-05-19 19:57:04 +00:00
|
|
|
t = get_arg(req, "t", "").strip()
|
2009-07-20 03:48:31 +00:00
|
|
|
replace = parse_replace_arg(get_arg(req, "replace", "true"))
|
|
|
|
|
2008-05-19 19:57:04 +00:00
|
|
|
assert self.parentnode and self.name
|
2008-10-28 20:41:04 +00:00
|
|
|
if req.getHeader("content-range"):
|
|
|
|
raise WebError("Content-Range in PUT not yet supported",
|
|
|
|
http.NOT_IMPLEMENTED)
|
2008-05-19 19:57:04 +00:00
|
|
|
if not t:
|
2009-02-20 19:15:54 +00:00
|
|
|
return self.replace_me_with_a_child(req, self.client, replace)
|
2008-05-19 19:57:04 +00:00
|
|
|
if t == "uri":
|
2009-02-20 19:15:54 +00:00
|
|
|
return self.replace_me_with_a_childcap(req, self.client, replace)
|
2008-05-19 19:57:04 +00:00
|
|
|
|
|
|
|
raise WebError("PUT to a file: bad t=%s" % t)
|
|
|
|
|
2019-09-10 23:52:20 +00:00
|
|
|
def render_POST(self, req):
|
2008-05-19 19:57:04 +00:00
|
|
|
t = get_arg(req, "t", "").strip()
|
|
|
|
replace = boolean_of_arg(get_arg(req, "replace", "true"))
|
2008-05-20 18:13:12 +00:00
|
|
|
if t == "upload":
|
2008-05-19 19:57:04 +00:00
|
|
|
# like PUT, but get the file data from an HTML form's input field.
|
|
|
|
# We could get here from POST /uri/mutablefilecap?t=upload,
|
|
|
|
# or POST /uri/path/file?t=upload, or
|
|
|
|
# POST /uri/path/dir?t=upload&name=foo . All have the same
|
|
|
|
# behavior, we just ignore any name= argument
|
2009-02-20 19:15:54 +00:00
|
|
|
d = self.replace_me_with_a_formpost(req, self.client, replace)
|
2008-05-19 19:57:04 +00:00
|
|
|
else:
|
2008-05-20 18:13:12 +00:00
|
|
|
# t=mkdir is handled in DirectoryNodeHandler._POST_mkdir, so
|
|
|
|
# there are no other t= values left to be handled by the
|
|
|
|
# placeholder.
|
2008-05-19 19:57:04 +00:00
|
|
|
raise WebError("POST to a file: bad t=%s" % t)
|
|
|
|
|
|
|
|
when_done = get_arg(req, "when_done", None)
|
|
|
|
if when_done:
|
2019-09-10 23:52:20 +00:00
|
|
|
d.addCallback(lambda res: when_done)
|
2008-05-19 19:57:04 +00:00
|
|
|
return d
|
|
|
|
|
|
|
|
|
2019-09-10 23:52:20 +00:00
|
|
|
class FileNodeHandler(Resource, ReplaceMeMixin, object):
|
2009-02-20 19:15:54 +00:00
|
|
|
def __init__(self, client, node, parentnode=None, name=None):
|
2019-09-10 23:52:20 +00:00
|
|
|
super(FileNodeHandler, self).__init__()
|
2009-02-20 19:15:54 +00:00
|
|
|
self.client = client
|
2008-05-19 19:57:04 +00:00
|
|
|
assert node
|
|
|
|
self.node = node
|
|
|
|
self.parentnode = parentnode
|
|
|
|
self.name = name
|
|
|
|
|
2019-09-10 23:52:20 +00:00
|
|
|
# XXX fixme, this was a childFactory override before, but it seems
|
|
|
|
# like it never does anything except raise errors? so .. why?
|
|
|
|
def getChild(self, name, req):
|
2011-08-24 15:59:28 +00:00
|
|
|
if isinstance(self.node, ProhibitedNode):
|
|
|
|
raise FileProhibited(self.node.reason)
|
2008-05-19 19:57:04 +00:00
|
|
|
if should_create_intermediate_directories(req):
|
2019-09-10 23:52:20 +00:00
|
|
|
raise WebError(
|
|
|
|
u"Cannot create directory {}, because its "
|
|
|
|
u"parent is a file, not a directory".format(name)
|
|
|
|
)
|
|
|
|
raise WebError(
|
|
|
|
"Files have no children named '{}'".format(
|
|
|
|
quote_output(name, encoding='utf-8'),
|
|
|
|
)
|
|
|
|
)
|
|
|
|
|
|
|
|
def render_GET(self, req):
|
2008-05-19 19:57:04 +00:00
|
|
|
t = get_arg(req, "t", "").strip()
|
2012-03-31 20:46:34 +00:00
|
|
|
|
2012-05-13 07:41:53 +00:00
|
|
|
# t=info contains variable ophandles, so is not allowed an ETag.
|
|
|
|
FIXED_OUTPUT_TYPES = ["", "json", "uri", "readonly-uri"]
|
|
|
|
if not self.node.is_mutable() and t in FIXED_OUTPUT_TYPES:
|
2012-03-31 20:46:34 +00:00
|
|
|
# if the client already has the ETag then we can
|
|
|
|
# short-circuit the whole process.
|
|
|
|
si = self.node.get_storage_index()
|
|
|
|
if si and req.setETag('%s-%s' % (base32.b2a(si), t or "")):
|
|
|
|
return ""
|
|
|
|
|
2008-05-19 19:57:04 +00:00
|
|
|
if not t:
|
|
|
|
# just get the contents
|
2008-07-19 01:06:50 +00:00
|
|
|
# the filename arrives as part of the URL or in a form input
|
|
|
|
# element, and will be sent back in a Content-Disposition header.
|
|
|
|
# Different browsers use various character sets for this name,
|
|
|
|
# sometimes depending upon how language environment is
|
|
|
|
# configured. Firefox sends the equivalent of
|
|
|
|
# urllib.quote(name.encode("utf-8")), while IE7 sometimes does
|
|
|
|
# latin-1. Browsers cannot agree on how to interpret the name
|
|
|
|
# they see in the Content-Disposition header either, despite some
|
|
|
|
# 11-year old standards (RFC2231) that explain how to do it
|
|
|
|
# properly. So we assume that at least the browser will agree
|
|
|
|
# with itself, and echo back the same bytes that we were given.
|
|
|
|
filename = get_arg(req, "filename", self.name) or "unknown"
|
2011-08-07 00:43:48 +00:00
|
|
|
d = self.node.get_best_readable_version()
|
2008-10-28 20:41:04 +00:00
|
|
|
d.addCallback(lambda dn: FileDownloader(dn, filename))
|
|
|
|
return d
|
2008-05-19 19:57:04 +00:00
|
|
|
if t == "json":
|
2011-08-07 00:43:48 +00:00
|
|
|
# We do this to make sure that fields like size and
|
|
|
|
# mutable-type (which depend on the file on the grid and not
|
|
|
|
# just on the cap) are filled in. The latter gets used in
|
|
|
|
# tests, in particular.
|
|
|
|
#
|
|
|
|
# TODO: Make it so that the servermap knows how to update in
|
|
|
|
# a mode specifically designed to fill in these fields, and
|
|
|
|
# then update it in that mode.
|
|
|
|
if self.node.is_mutable():
|
|
|
|
d = self.node.get_servermap(MODE_READ)
|
2009-12-27 22:54:43 +00:00
|
|
|
else:
|
|
|
|
d = defer.succeed(None)
|
2011-08-07 00:43:48 +00:00
|
|
|
if self.parentnode and self.name:
|
|
|
|
d.addCallback(lambda ignored:
|
|
|
|
self.parentnode.get_metadata_for(self.name))
|
|
|
|
else:
|
|
|
|
d.addCallback(lambda ignored: None)
|
2019-09-11 00:47:02 +00:00
|
|
|
d.addCallback(lambda md: FileJSONMetadata(req, self.node, md))
|
2009-12-27 22:54:43 +00:00
|
|
|
return d
|
2008-09-18 05:00:41 +00:00
|
|
|
if t == "info":
|
|
|
|
return MoreInfo(self.node)
|
2008-05-19 19:57:04 +00:00
|
|
|
if t == "uri":
|
2019-09-11 00:47:02 +00:00
|
|
|
return FileURI(req, self.node)
|
2008-05-19 19:57:04 +00:00
|
|
|
if t == "readonly-uri":
|
2019-09-11 00:47:02 +00:00
|
|
|
return FileReadOnlyURI(req, self.node)
|
2008-05-19 19:57:04 +00:00
|
|
|
raise WebError("GET file: bad t=%s" % t)
|
|
|
|
|
2019-09-10 23:52:20 +00:00
|
|
|
def render_HEAD(self, req):
|
2008-05-19 19:57:04 +00:00
|
|
|
t = get_arg(req, "t", "").strip()
|
|
|
|
if t:
|
2012-03-31 20:46:34 +00:00
|
|
|
raise WebError("HEAD file: bad t=%s" % t)
|
2008-08-13 02:04:10 +00:00
|
|
|
filename = get_arg(req, "filename", self.name) or "unknown"
|
2011-08-07 00:43:48 +00:00
|
|
|
d = self.node.get_best_readable_version()
|
2008-10-28 20:41:04 +00:00
|
|
|
d.addCallback(lambda dn: FileDownloader(dn, filename))
|
2008-05-19 19:57:04 +00:00
|
|
|
return d
|
|
|
|
|
2019-09-10 23:52:20 +00:00
|
|
|
def render_PUT(self, req):
|
2008-05-19 19:57:04 +00:00
|
|
|
t = get_arg(req, "t", "").strip()
|
2009-07-20 03:48:31 +00:00
|
|
|
replace = parse_replace_arg(get_arg(req, "replace", "true"))
|
2011-08-07 00:43:48 +00:00
|
|
|
offset = parse_offset_arg(get_arg(req, "offset", None))
|
2009-07-20 03:48:31 +00:00
|
|
|
|
2008-05-19 19:57:04 +00:00
|
|
|
if not t:
|
|
|
|
if not replace:
|
|
|
|
# this is the early trap: if someone else modifies the
|
|
|
|
# directory while we're uploading, the add_file(overwrite=)
|
|
|
|
# call in replace_me_with_a_child will do the late trap.
|
|
|
|
raise ExistingChildError()
|
2011-08-07 00:43:48 +00:00
|
|
|
|
|
|
|
if self.node.is_mutable():
|
|
|
|
# Are we a readonly filenode? We shouldn't allow callers
|
|
|
|
# to try to replace us if we are.
|
|
|
|
if self.node.is_readonly():
|
|
|
|
raise WebError("PUT to a mutable file: replace or update"
|
|
|
|
" requested with read-only cap")
|
|
|
|
if offset is None:
|
|
|
|
return self.replace_my_contents(req)
|
|
|
|
|
|
|
|
if offset >= 0:
|
|
|
|
return self.update_my_contents(req, offset)
|
|
|
|
|
|
|
|
raise WebError("PUT to a mutable file: Invalid offset")
|
|
|
|
|
|
|
|
else:
|
|
|
|
if offset is not None:
|
|
|
|
raise WebError("PUT to a file: append operation invoked "
|
|
|
|
"on an immutable cap")
|
|
|
|
|
|
|
|
assert self.parentnode and self.name
|
|
|
|
return self.replace_me_with_a_child(req, self.client, replace)
|
|
|
|
|
2008-05-19 19:57:04 +00:00
|
|
|
if t == "uri":
|
|
|
|
if not replace:
|
|
|
|
raise ExistingChildError()
|
|
|
|
assert self.parentnode and self.name
|
2009-02-20 19:15:54 +00:00
|
|
|
return self.replace_me_with_a_childcap(req, self.client, replace)
|
2008-05-19 19:57:04 +00:00
|
|
|
|
|
|
|
raise WebError("PUT to a file: bad t=%s" % t)
|
|
|
|
|
2019-09-10 23:52:20 +00:00
|
|
|
def render_POST(self, req):
|
2008-05-19 19:57:04 +00:00
|
|
|
t = get_arg(req, "t", "").strip()
|
|
|
|
replace = boolean_of_arg(get_arg(req, "replace", "true"))
|
|
|
|
if t == "check":
|
|
|
|
d = self._POST_check(req)
|
|
|
|
elif t == "upload":
|
|
|
|
# like PUT, but get the file data from an HTML form's input field
|
|
|
|
# We could get here from POST /uri/mutablefilecap?t=upload,
|
|
|
|
# or POST /uri/path/file?t=upload, or
|
|
|
|
# POST /uri/path/dir?t=upload&name=foo . All have the same
|
|
|
|
# behavior, we just ignore any name= argument
|
|
|
|
if self.node.is_mutable():
|
2009-02-20 19:15:54 +00:00
|
|
|
d = self.replace_my_contents_with_a_formpost(req)
|
2008-05-19 19:57:04 +00:00
|
|
|
else:
|
|
|
|
if not replace:
|
|
|
|
raise ExistingChildError()
|
|
|
|
assert self.parentnode and self.name
|
2009-02-20 19:15:54 +00:00
|
|
|
d = self.replace_me_with_a_formpost(req, self.client, replace)
|
2008-05-19 19:57:04 +00:00
|
|
|
else:
|
|
|
|
raise WebError("POST to file: bad t=%s" % t)
|
|
|
|
|
|
|
|
when_done = get_arg(req, "when_done", None)
|
|
|
|
if when_done:
|
|
|
|
d.addCallback(lambda res: url.URL.fromString(when_done))
|
|
|
|
return d
|
|
|
|
|
2009-11-26 23:27:31 +00:00
|
|
|
def _maybe_literal(self, res, Results_Class):
|
|
|
|
if res:
|
|
|
|
return Results_Class(self.client, res)
|
2012-04-03 03:02:59 +00:00
|
|
|
return LiteralCheckResultsRenderer(self.client)
|
2009-11-26 23:27:31 +00:00
|
|
|
|
2008-05-19 19:57:04 +00:00
|
|
|
def _POST_check(self, req):
|
2008-08-13 00:05:01 +00:00
|
|
|
verify = boolean_of_arg(get_arg(req, "verify", "false"))
|
|
|
|
repair = boolean_of_arg(get_arg(req, "repair", "false"))
|
2009-02-18 02:32:43 +00:00
|
|
|
add_lease = boolean_of_arg(get_arg(req, "add-lease", "false"))
|
2008-09-07 19:44:56 +00:00
|
|
|
if repair:
|
2009-02-18 02:32:43 +00:00
|
|
|
d = self.node.check_and_repair(Monitor(), verify, add_lease)
|
2012-04-03 03:02:59 +00:00
|
|
|
d.addCallback(self._maybe_literal, CheckAndRepairResultsRenderer)
|
2008-09-07 19:44:56 +00:00
|
|
|
else:
|
2009-02-18 02:32:43 +00:00
|
|
|
d = self.node.check(Monitor(), verify, add_lease)
|
2012-04-03 03:02:59 +00:00
|
|
|
d.addCallback(self._maybe_literal, CheckResultsRenderer)
|
2008-05-19 19:57:04 +00:00
|
|
|
return d
|
|
|
|
|
2019-09-11 00:47:02 +00:00
|
|
|
def render_DELETE(self, req):
|
2008-05-19 19:57:04 +00:00
|
|
|
assert self.parentnode and self.name
|
|
|
|
d = self.parentnode.delete(self.name)
|
|
|
|
d.addCallback(lambda res: self.node.get_uri())
|
|
|
|
return d
|
|
|
|
|
2009-02-20 19:15:54 +00:00
|
|
|
def replace_my_contents(self, req):
|
2008-05-19 19:57:04 +00:00
|
|
|
req.content.seek(0)
|
2011-08-07 00:43:48 +00:00
|
|
|
new_contents = MutableFileHandle(req.content)
|
2008-05-19 19:57:04 +00:00
|
|
|
d = self.node.overwrite(new_contents)
|
|
|
|
d.addCallback(lambda res: self.node.get_uri())
|
|
|
|
return d
|
|
|
|
|
2011-08-07 00:43:48 +00:00
|
|
|
|
|
|
|
def update_my_contents(self, req, offset):
|
|
|
|
req.content.seek(0)
|
|
|
|
added_contents = MutableFileHandle(req.content)
|
|
|
|
|
|
|
|
d = self.node.get_best_mutable_version()
|
|
|
|
d.addCallback(lambda mv:
|
|
|
|
mv.update(added_contents, offset))
|
|
|
|
d.addCallback(lambda ignored:
|
|
|
|
self.node.get_uri())
|
|
|
|
return d
|
|
|
|
|
|
|
|
|
2009-02-20 19:15:54 +00:00
|
|
|
def replace_my_contents_with_a_formpost(self, req):
|
2008-05-19 19:57:04 +00:00
|
|
|
# we have a mutable file. Get the data from the formpost, and replace
|
|
|
|
# the mutable file's contents with it.
|
2011-08-07 00:43:48 +00:00
|
|
|
new_contents = req.fields['file']
|
|
|
|
new_contents = MutableFileHandle(new_contents.file)
|
|
|
|
|
2008-05-19 19:57:04 +00:00
|
|
|
d = self.node.overwrite(new_contents)
|
|
|
|
d.addCallback(lambda res: self.node.get_uri())
|
|
|
|
return d
|
|
|
|
|
2008-10-28 20:41:04 +00:00
|
|
|
|
2019-09-10 23:52:20 +00:00
|
|
|
class FileDownloader(Resource, object):
|
2008-10-28 20:41:04 +00:00
|
|
|
def __init__(self, filenode, filename):
|
2019-09-10 23:52:20 +00:00
|
|
|
super(FileDownloader, self).__init__()
|
2008-05-19 19:57:04 +00:00
|
|
|
self.filenode = filenode
|
|
|
|
self.filename = filename
|
2008-10-28 20:41:04 +00:00
|
|
|
|
2010-03-10 03:59:13 +00:00
|
|
|
def parse_range_header(self, range):
|
|
|
|
# Parse a byte ranges according to RFC 2616 "14.35.1 Byte
|
|
|
|
# Ranges". Returns None if the range doesn't make sense so it
|
|
|
|
# can be ignored (per the spec). When successful, returns a
|
|
|
|
# list of (first,last) inclusive range tuples.
|
|
|
|
|
|
|
|
filesize = self.filenode.get_size()
|
|
|
|
assert isinstance(filesize, (int,long)), filesize
|
|
|
|
|
|
|
|
try:
|
|
|
|
# byte-ranges-specifier
|
|
|
|
units, rangeset = range.split('=', 1)
|
|
|
|
if units != 'bytes':
|
|
|
|
return None # nothing else supported
|
|
|
|
|
|
|
|
def parse_range(r):
|
|
|
|
first, last = r.split('-', 1)
|
|
|
|
|
2019-01-24 14:53:02 +00:00
|
|
|
if first == '':
|
2010-03-10 03:59:13 +00:00
|
|
|
# suffix-byte-range-spec
|
|
|
|
first = filesize - long(last)
|
|
|
|
last = filesize - 1
|
|
|
|
else:
|
|
|
|
# byte-range-spec
|
|
|
|
|
|
|
|
# first-byte-pos
|
|
|
|
first = long(first)
|
|
|
|
|
|
|
|
# last-byte-pos
|
2019-01-24 14:53:02 +00:00
|
|
|
if last == '':
|
2010-03-10 03:59:13 +00:00
|
|
|
last = filesize - 1
|
|
|
|
else:
|
|
|
|
last = long(last)
|
|
|
|
|
|
|
|
if last < first:
|
|
|
|
raise ValueError
|
|
|
|
|
|
|
|
return (first, last)
|
|
|
|
|
|
|
|
# byte-range-set
|
|
|
|
#
|
|
|
|
# Note: the spec uses "1#" for the list of ranges, which
|
|
|
|
# implicitly allows whitespace around the ',' separators,
|
|
|
|
# so strip it.
|
|
|
|
return [ parse_range(r.strip()) for r in rangeset.split(',') ]
|
|
|
|
except ValueError:
|
|
|
|
return None
|
|
|
|
|
2019-09-10 23:52:20 +00:00
|
|
|
def render(self, req):
|
2008-05-19 19:57:04 +00:00
|
|
|
gte = static.getTypeAndEncoding
|
|
|
|
ctype, encoding = gte(self.filename,
|
|
|
|
static.File.contentTypes,
|
|
|
|
static.File.contentEncodings,
|
|
|
|
defaultType="text/plain")
|
2008-10-28 20:41:04 +00:00
|
|
|
req.setHeader("content-type", ctype)
|
|
|
|
if encoding:
|
|
|
|
req.setHeader("content-encoding", encoding)
|
|
|
|
|
|
|
|
if boolean_of_arg(get_arg(req, "save", "False")):
|
|
|
|
# tell the browser to save the file rather display it we don't
|
|
|
|
# try to encode the filename, instead we echo back the exact same
|
|
|
|
# bytes we were given in the URL. See the comment in
|
|
|
|
# FileNodeHandler.render_GET for the sad details.
|
|
|
|
req.setHeader("content-disposition",
|
|
|
|
'attachment; filename="%s"' % self.filename)
|
|
|
|
|
|
|
|
filesize = self.filenode.get_size()
|
|
|
|
assert isinstance(filesize, (int,long)), filesize
|
2010-03-10 03:59:13 +00:00
|
|
|
first, size = 0, None
|
2008-10-28 20:41:04 +00:00
|
|
|
contentsize = filesize
|
2008-10-29 01:01:03 +00:00
|
|
|
req.setHeader("accept-ranges", "bytes")
|
2012-05-13 03:42:52 +00:00
|
|
|
|
2008-10-29 01:17:20 +00:00
|
|
|
# TODO: for mutable files, use the roothash. For LIT, hash the data.
|
|
|
|
# or maybe just use the URI for CHK and LIT.
|
2008-10-28 20:41:04 +00:00
|
|
|
rangeheader = req.getHeader('range')
|
|
|
|
if rangeheader:
|
2010-03-10 03:59:13 +00:00
|
|
|
ranges = self.parse_range_header(rangeheader)
|
|
|
|
|
|
|
|
# ranges = None means the header didn't parse, so ignore
|
|
|
|
# the header as if it didn't exist. If is more than one
|
|
|
|
# range, then just return the first for now, until we can
|
|
|
|
# generate multipart/byteranges.
|
|
|
|
if ranges is not None:
|
|
|
|
first, last = ranges[0]
|
|
|
|
|
|
|
|
if first >= filesize:
|
|
|
|
raise WebError('First beyond end of file',
|
|
|
|
http.REQUESTED_RANGE_NOT_SATISFIABLE)
|
|
|
|
else:
|
|
|
|
first = max(0, first)
|
|
|
|
last = min(filesize-1, last)
|
|
|
|
|
|
|
|
req.setResponseCode(http.PARTIAL_CONTENT)
|
|
|
|
req.setHeader('content-range',"bytes %s-%s/%s" %
|
|
|
|
(str(first), str(last),
|
|
|
|
str(filesize)))
|
|
|
|
contentsize = last - first + 1
|
|
|
|
size = contentsize
|
|
|
|
|
2015-01-21 18:31:31 +00:00
|
|
|
req.setHeader("content-length", b"%d" % contentsize)
|
2008-10-28 20:41:04 +00:00
|
|
|
if req.method == "HEAD":
|
|
|
|
return ""
|
2011-02-21 06:15:44 +00:00
|
|
|
|
|
|
|
finished = []
|
|
|
|
def _request_finished(ign):
|
|
|
|
finished.append(True)
|
2011-10-13 20:12:19 +00:00
|
|
|
req.notifyFinish().addBoth(_request_finished)
|
2011-02-21 06:15:44 +00:00
|
|
|
|
2010-03-10 03:59:13 +00:00
|
|
|
d = self.filenode.read(req, first, size)
|
2011-02-21 06:15:44 +00:00
|
|
|
|
|
|
|
def _finished(ign):
|
|
|
|
if not finished:
|
|
|
|
req.finish()
|
2008-10-28 20:41:04 +00:00
|
|
|
def _error(f):
|
2011-02-21 06:15:44 +00:00
|
|
|
lp = log.msg("error during GET", facility="tahoe.webish", failure=f,
|
|
|
|
level=log.UNUSUAL, umid="xSiF3w")
|
|
|
|
if finished:
|
|
|
|
log.msg("but it's too late to tell them", parent=lp,
|
|
|
|
level=log.UNUSUAL, umid="j1xIbw")
|
|
|
|
return
|
2010-08-06 07:07:05 +00:00
|
|
|
req._tahoe_request_had_error = f # for HTTP-style logging
|
2008-10-28 20:41:04 +00:00
|
|
|
if req.startedWriting:
|
|
|
|
# The content-type is already set, and the response code has
|
|
|
|
# already been sent, so we can't provide a clean error
|
|
|
|
# indication. We can emit text (which a browser might
|
|
|
|
# interpret as something else), and if we sent a Size header,
|
|
|
|
# they might notice that we've truncated the data. Keep the
|
|
|
|
# error message small to improve the chances of having our
|
|
|
|
# error response be shorter than the intended results.
|
|
|
|
#
|
|
|
|
# We don't have a lot of options, unfortunately.
|
|
|
|
req.write("problem during download\n")
|
2009-03-04 04:56:30 +00:00
|
|
|
req.finish()
|
2008-10-28 20:41:04 +00:00
|
|
|
else:
|
|
|
|
# We haven't written anything yet, so we can provide a
|
|
|
|
# sensible error message.
|
2009-03-04 04:56:30 +00:00
|
|
|
eh = MyExceptionHandler()
|
2019-09-11 00:47:02 +00:00
|
|
|
eh.renderHTTP_exception(req, f)
|
2011-02-21 06:15:44 +00:00
|
|
|
d.addCallbacks(_finished, _error)
|
2008-10-28 20:41:04 +00:00
|
|
|
return req.deferred
|
|
|
|
|
2008-05-19 19:57:04 +00:00
|
|
|
|
2019-09-11 00:47:02 +00:00
|
|
|
def FileJSONMetadata(req, filenode, edge_metadata):
|
2010-01-27 06:44:30 +00:00
|
|
|
rw_uri = filenode.get_write_uri()
|
|
|
|
ro_uri = filenode.get_readonly_uri()
|
2014-10-07 19:11:31 +00:00
|
|
|
data = ("filenode", get_filenode_metadata(filenode))
|
2008-05-20 22:14:19 +00:00
|
|
|
if ro_uri:
|
|
|
|
data[1]['ro_uri'] = ro_uri
|
|
|
|
if rw_uri:
|
|
|
|
data[1]['rw_uri'] = rw_uri
|
2009-02-04 02:22:48 +00:00
|
|
|
verifycap = filenode.get_verify_cap()
|
|
|
|
if verifycap:
|
|
|
|
data[1]['verify_uri'] = verifycap.to_string()
|
2009-12-27 23:21:49 +00:00
|
|
|
if edge_metadata is not None:
|
|
|
|
data[1]['metadata'] = edge_metadata
|
2011-08-07 00:43:48 +00:00
|
|
|
|
2019-09-11 00:47:02 +00:00
|
|
|
return text_plain(json.dumps(data, indent=1) + "\n", req)
|
2008-05-19 19:57:04 +00:00
|
|
|
|
2019-09-11 00:47:02 +00:00
|
|
|
def FileURI(req, filenode):
|
|
|
|
return text_plain(filenode.get_uri(), req)
|
2008-05-19 19:57:04 +00:00
|
|
|
|
2019-09-11 00:47:02 +00:00
|
|
|
def FileReadOnlyURI(req, filenode):
|
2008-05-19 19:57:04 +00:00
|
|
|
if filenode.is_readonly():
|
2019-09-11 00:47:02 +00:00
|
|
|
return text_plain(filenode.get_uri(), req)
|
|
|
|
return text_plain(filenode.get_readonly_uri(), req)
|
2008-05-19 19:57:04 +00:00
|
|
|
|
|
|
|
class FileNodeDownloadHandler(FileNodeHandler):
|
2019-09-11 00:47:02 +00:00
|
|
|
def childFactory(self, req, name):
|
2009-02-20 19:15:54 +00:00
|
|
|
return FileNodeDownloadHandler(self.client, self.node, name=name)
|