2006-12-04 02:07:41 +00:00
|
|
|
|
|
|
|
"""This is the client-side facility to manipulate virtual drives."""
|
|
|
|
|
2007-06-25 20:23:51 +00:00
|
|
|
import os.path
|
|
|
|
from zope.interface import implements
|
2006-12-04 02:07:41 +00:00
|
|
|
from twisted.application import service
|
|
|
|
from twisted.internet import defer
|
2006-12-04 07:46:36 +00:00
|
|
|
from twisted.python import log
|
2007-06-25 20:23:51 +00:00
|
|
|
from allmydata import upload, download, uri
|
|
|
|
from allmydata.Crypto.Cipher import AES
|
|
|
|
from allmydata.util import hashutil, idlib
|
|
|
|
from allmydata.interfaces import IDirectoryNode, IFileNode
|
|
|
|
|
|
|
|
class NotMutableError(Exception):
|
|
|
|
pass
|
2006-12-04 02:07:41 +00:00
|
|
|
|
|
|
|
class VDrive(service.MultiService):
|
|
|
|
name = "vdrive"
|
|
|
|
|
2007-06-15 03:14:34 +00:00
|
|
|
def set_server(self, vdrive_server):
|
|
|
|
self.gvd_server = vdrive_server
|
2006-12-04 02:07:41 +00:00
|
|
|
def set_root(self, root):
|
|
|
|
self.gvd_root = root
|
|
|
|
|
|
|
|
def dirpath(self, dir_or_path):
|
|
|
|
if isinstance(dir_or_path, str):
|
|
|
|
return self.get_dir(dir_or_path)
|
|
|
|
return defer.succeed(dir_or_path)
|
|
|
|
|
|
|
|
def get_dir(self, path):
|
|
|
|
"""Return a Deferred that fires with a RemoteReference to a
|
|
|
|
MutableDirectoryNode at the given /-delimited path."""
|
|
|
|
d = defer.succeed(self.gvd_root)
|
|
|
|
if path.startswith("/"):
|
|
|
|
path = path[1:]
|
|
|
|
if path == "":
|
|
|
|
return d
|
|
|
|
for piece in path.split("/"):
|
|
|
|
d.addCallback(lambda parent: parent.callRemote("list"))
|
|
|
|
def _find(table, subdir):
|
|
|
|
for name,target in table:
|
|
|
|
if name == subdir:
|
2006-12-04 04:11:26 +00:00
|
|
|
return target
|
2006-12-04 02:07:41 +00:00
|
|
|
else:
|
|
|
|
raise KeyError("no such directory '%s' in '%s'" %
|
|
|
|
(subdir, [t[0] for t in table]))
|
|
|
|
d.addCallback(_find, piece)
|
|
|
|
def _check(subdir):
|
2006-12-04 04:11:26 +00:00
|
|
|
assert not isinstance(subdir, str), "Hey, %s shouldn't be a string" % subdir
|
2006-12-04 02:07:41 +00:00
|
|
|
return subdir
|
|
|
|
d.addCallback(_check)
|
|
|
|
return d
|
|
|
|
|
2007-01-17 02:43:13 +00:00
|
|
|
def get_uri_from_parent(self, parent, filename):
|
2006-12-04 05:42:19 +00:00
|
|
|
assert not isinstance(parent, str), "'%s' isn't a directory node" % (parent,)
|
|
|
|
d = parent.callRemote("list")
|
|
|
|
def _find(table):
|
|
|
|
for name,target in table:
|
|
|
|
if name == filename:
|
|
|
|
assert isinstance(target, str), "Hey, %s isn't a file" % filename
|
|
|
|
return target
|
|
|
|
else:
|
|
|
|
raise KeyError("no such file '%s' in '%s'" %
|
|
|
|
(filename, [t[0] for t in table]))
|
|
|
|
d.addCallback(_find)
|
|
|
|
return d
|
|
|
|
|
2006-12-04 02:07:41 +00:00
|
|
|
def get_root(self):
|
|
|
|
return self.gvd_root
|
|
|
|
|
|
|
|
def listdir(self, dir_or_path):
|
|
|
|
d = self.dirpath(dir_or_path)
|
|
|
|
d.addCallback(lambda parent: parent.callRemote("list"))
|
|
|
|
def _list(table):
|
|
|
|
return [t[0] for t in table]
|
|
|
|
d.addCallback(_list)
|
|
|
|
return d
|
|
|
|
|
|
|
|
def put_file(self, dir_or_path, name, uploadable):
|
|
|
|
"""Upload an IUploadable and add it to the virtual drive (as an entry
|
|
|
|
called 'name', in 'dir_or_path') 'dir_or_path' must either be a
|
|
|
|
string like 'root/subdir1/subdir2', or a directory node (either the
|
|
|
|
root directory node returned by get_root(), or a subdirectory
|
|
|
|
returned by list() ).
|
|
|
|
|
|
|
|
The uploadable can be an instance of allmydata.upload.Data,
|
|
|
|
FileHandle, or FileName.
|
|
|
|
|
|
|
|
I return a deferred that will fire when the operation is complete.
|
|
|
|
"""
|
|
|
|
|
2006-12-04 07:46:36 +00:00
|
|
|
log.msg("putting file to '%s'" % name)
|
2006-12-04 05:42:19 +00:00
|
|
|
ul = self.parent.getServiceNamed("uploader")
|
2006-12-04 02:07:41 +00:00
|
|
|
d = self.dirpath(dir_or_path)
|
|
|
|
def _got_dir(dirnode):
|
2006-12-04 05:42:19 +00:00
|
|
|
d1 = ul.upload(uploadable)
|
2007-01-16 04:22:22 +00:00
|
|
|
def _add(uri):
|
2007-04-24 08:41:40 +00:00
|
|
|
d2 = dirnode.callRemote("add_file", name, uri)
|
|
|
|
d2.addCallback(lambda res: uri)
|
|
|
|
return d2
|
2007-01-16 04:22:22 +00:00
|
|
|
d1.addCallback(_add)
|
2006-12-04 02:07:41 +00:00
|
|
|
return d1
|
|
|
|
d.addCallback(_got_dir)
|
2006-12-04 07:46:36 +00:00
|
|
|
def _done(res):
|
|
|
|
log.msg("finished putting file to '%s'" % name)
|
|
|
|
return res
|
|
|
|
d.addCallback(_done)
|
2006-12-04 02:07:41 +00:00
|
|
|
return d
|
|
|
|
|
|
|
|
def put_file_by_filename(self, dir_or_path, name, filename):
|
2006-12-04 05:42:19 +00:00
|
|
|
return self.put_file(dir_or_path, name, upload.FileName(filename))
|
2006-12-04 02:07:41 +00:00
|
|
|
def put_file_by_data(self, dir_or_path, name, data):
|
2006-12-04 05:42:19 +00:00
|
|
|
return self.put_file(dir_or_path, name, upload.Data(data))
|
2006-12-04 02:07:41 +00:00
|
|
|
def put_file_by_filehandle(self, dir_or_path, name, filehandle):
|
2006-12-04 05:42:19 +00:00
|
|
|
return self.put_file(dir_or_path, name, upload.FileHandle(filehandle))
|
2006-12-04 02:07:41 +00:00
|
|
|
|
|
|
|
def make_directory(self, dir_or_path, name):
|
|
|
|
d = self.dirpath(dir_or_path)
|
|
|
|
d.addCallback(lambda parent: parent.callRemote("add_directory", name))
|
|
|
|
return d
|
|
|
|
|
2006-12-05 02:27:38 +00:00
|
|
|
def remove(self, parent, name):
|
|
|
|
assert not isinstance(parent, str)
|
|
|
|
log.msg("vdrive removing %s" % name)
|
2007-01-17 02:43:13 +00:00
|
|
|
# first find the uri
|
|
|
|
d = self.get_uri_from_parent(parent, name)
|
|
|
|
def _got_uri(vid):
|
2006-12-05 02:27:38 +00:00
|
|
|
# TODO: delete the file's shares using this
|
|
|
|
pass
|
2007-01-17 02:43:13 +00:00
|
|
|
d.addCallback(_got_uri)
|
2006-12-05 02:27:38 +00:00
|
|
|
def _delete_from_parent(res):
|
|
|
|
return parent.callRemote("remove", name)
|
|
|
|
d.addCallback(_delete_from_parent)
|
|
|
|
def _done(res):
|
|
|
|
log.msg("vdrive done removing %s" % name)
|
|
|
|
d.addCallback(_done)
|
|
|
|
return d
|
|
|
|
|
2006-12-04 05:42:19 +00:00
|
|
|
|
|
|
|
def get_file(self, dir_and_name_or_path, download_target):
|
|
|
|
"""Retrieve a file from the virtual drive and put it somewhere.
|
|
|
|
|
|
|
|
The file to be retrieved may either be specified as a (dir, name)
|
|
|
|
tuple or as a full /-delimited pathname. In the former case, 'dir'
|
|
|
|
can be either a DirectoryNode or a pathname.
|
|
|
|
|
|
|
|
The download target must be an IDownloadTarget instance like
|
|
|
|
allmydata.download.Data, .FileName, or .FileHandle .
|
|
|
|
"""
|
|
|
|
|
2006-12-04 07:46:36 +00:00
|
|
|
log.msg("getting file from %s" % (dir_and_name_or_path,))
|
2006-12-04 05:42:19 +00:00
|
|
|
dl = self.parent.getServiceNamed("downloader")
|
|
|
|
|
|
|
|
if isinstance(dir_and_name_or_path, tuple):
|
|
|
|
dir_or_path, name = dir_and_name_or_path
|
|
|
|
d = self.dirpath(dir_or_path)
|
|
|
|
def _got_dir(dirnode):
|
2007-01-17 02:43:13 +00:00
|
|
|
return self.get_uri_from_parent(dirnode, name)
|
2006-12-04 05:42:19 +00:00
|
|
|
d.addCallback(_got_dir)
|
|
|
|
else:
|
|
|
|
rslash = dir_and_name_or_path.rfind("/")
|
|
|
|
if rslash == -1:
|
|
|
|
# we're looking for a file in the root directory
|
|
|
|
dir = self.gvd_root
|
|
|
|
name = dir_and_name_or_path
|
2007-01-17 02:43:13 +00:00
|
|
|
d = self.get_uri_from_parent(dir, name)
|
2006-12-04 05:42:19 +00:00
|
|
|
else:
|
|
|
|
dirpath = dir_and_name_or_path[:rslash]
|
|
|
|
name = dir_and_name_or_path[rslash+1:]
|
|
|
|
d = self.dirpath(dirpath)
|
|
|
|
d.addCallback(lambda dir:
|
2007-01-17 02:43:13 +00:00
|
|
|
self.get_uri_from_parent(dir, name))
|
2006-12-04 05:42:19 +00:00
|
|
|
|
2007-01-17 02:43:13 +00:00
|
|
|
def _got_uri(uri):
|
|
|
|
return dl.download(uri, download_target)
|
|
|
|
d.addCallback(_got_uri)
|
2006-12-04 07:46:36 +00:00
|
|
|
def _done(res):
|
|
|
|
log.msg("finished getting file")
|
|
|
|
return res
|
|
|
|
d.addCallback(_done)
|
2006-12-04 05:42:19 +00:00
|
|
|
return d
|
|
|
|
|
|
|
|
def get_file_to_filename(self, from_where, filename):
|
|
|
|
return self.get_file(from_where, download.FileName(filename))
|
|
|
|
def get_file_to_data(self, from_where):
|
|
|
|
return self.get_file(from_where, download.Data())
|
|
|
|
def get_file_to_filehandle(self, from_where, filehandle):
|
|
|
|
return self.get_file(from_where, download.FileHandle(filehandle))
|
|
|
|
|
2007-06-15 03:14:34 +00:00
|
|
|
|
2007-06-25 20:23:51 +00:00
|
|
|
def create_directory_node(client, diruri):
|
|
|
|
assert uri.is_dirnode_uri(diruri)
|
|
|
|
if uri.is_mutable_dirnode_uri(diruri):
|
|
|
|
dirnode_class = MutableDirectoryNode
|
|
|
|
else:
|
|
|
|
dirnode_class = ImmutableDirectoryNode
|
|
|
|
(furl, key) = uri.unpack_dirnode_uri(diruri)
|
|
|
|
d = client.tub.getReference(furl)
|
|
|
|
def _got(rref):
|
|
|
|
dirnode = dirnode_class(diruri, client, rref, key)
|
|
|
|
return dirnode
|
|
|
|
d.addCallback(_got)
|
|
|
|
return d
|
|
|
|
|
|
|
|
def encrypt(key, data):
|
|
|
|
# TODO: add the hmac
|
|
|
|
IV = os.urandom(14)
|
|
|
|
counterstart = IV + "\x00"*2
|
|
|
|
assert len(counterstart) == 16, len(counterstart)
|
|
|
|
cryptor = AES.new(key=key, mode=AES.MODE_CTR, counterstart=counterstart)
|
|
|
|
crypttext = cryptor.encrypt(data)
|
|
|
|
return IV + crypttext
|
|
|
|
|
|
|
|
def decrypt(key, data):
|
|
|
|
# TODO: validate the hmac
|
|
|
|
assert len(data) >= 14, len(data)
|
|
|
|
IV = data[:14]
|
|
|
|
counterstart = IV + "\x00"*2
|
|
|
|
assert len(counterstart) == 16, len(counterstart)
|
|
|
|
cryptor = AES.new(key=key, mode=AES.MODE_CTR, counterstart=counterstart)
|
|
|
|
plaintext = cryptor.decrypt(data[14:])
|
|
|
|
return plaintext
|
|
|
|
|
|
|
|
|
|
|
|
class ImmutableDirectoryNode:
|
|
|
|
implements(IDirectoryNode)
|
|
|
|
|
|
|
|
def __init__(self, myuri, client, rref, readkey):
|
|
|
|
self._uri = myuri
|
2007-06-15 07:37:32 +00:00
|
|
|
self._client = client
|
2007-06-25 20:23:51 +00:00
|
|
|
self._tub = client.tub
|
|
|
|
self._rref = rref
|
|
|
|
self._readkey = readkey
|
|
|
|
self._writekey = None
|
|
|
|
self._write_enabler = None
|
|
|
|
self._index = hashutil.dir_index_hash(self._readkey)
|
|
|
|
self._mutable = False
|
|
|
|
|
|
|
|
def dump(self):
|
|
|
|
return ["URI: %s" % self._uri,
|
|
|
|
"rk: %s" % idlib.b2a(self._readkey),
|
|
|
|
"index: %s" % idlib.b2a(self._index),
|
|
|
|
]
|
|
|
|
|
|
|
|
def is_mutable(self):
|
|
|
|
return self._mutable
|
|
|
|
|
|
|
|
def get_uri(self):
|
|
|
|
return self._uri
|
|
|
|
|
|
|
|
def get_immutable_uri(self):
|
|
|
|
# return the dirnode URI for a read-only form of this directory
|
|
|
|
if self._mutable:
|
|
|
|
return uri.make_immutable_dirnode_uri(self._uri)
|
|
|
|
else:
|
|
|
|
return self._uri
|
|
|
|
|
2007-06-15 07:37:32 +00:00
|
|
|
def __hash__(self):
|
2007-06-25 20:23:51 +00:00
|
|
|
return hash((self.__class__, self._uri))
|
2007-06-15 07:37:32 +00:00
|
|
|
def __cmp__(self, them):
|
|
|
|
if cmp(type(self), type(them)):
|
|
|
|
return cmp(type(self), type(them))
|
|
|
|
if cmp(self.__class__, them.__class__):
|
|
|
|
return cmp(self.__class__, them.__class__)
|
2007-06-25 20:23:51 +00:00
|
|
|
return cmp(self._uri, them._uri)
|
|
|
|
|
|
|
|
def _encrypt(self, key, data):
|
|
|
|
return encrypt(key, data)
|
|
|
|
|
|
|
|
def _decrypt(self, key, data):
|
|
|
|
return decrypt(key, data)
|
|
|
|
|
|
|
|
def _decrypt_child(self, E_write, E_read):
|
|
|
|
if E_write and self._writekey:
|
|
|
|
# we prefer read-write children when we can get them
|
|
|
|
return self._decrypt(self._writekey, E_write)
|
|
|
|
else:
|
|
|
|
return self._decrypt(self._readkey, E_read)
|
2007-06-15 03:14:34 +00:00
|
|
|
|
2007-06-15 07:37:32 +00:00
|
|
|
def list(self):
|
2007-06-25 20:23:51 +00:00
|
|
|
d = self._rref.callRemote("list", self._index)
|
|
|
|
entries = {}
|
|
|
|
def _got(res):
|
|
|
|
dl = []
|
|
|
|
for (E_name, E_write, E_read) in res:
|
|
|
|
name = self._decrypt(self._readkey, E_name)
|
|
|
|
child_uri = self._decrypt_child(E_write, E_read)
|
|
|
|
d2 = self._create_node(child_uri)
|
|
|
|
def _created(node, name):
|
|
|
|
entries[name] = node
|
|
|
|
d2.addCallback(_created, name)
|
|
|
|
dl.append(d2)
|
|
|
|
return defer.DeferredList(dl)
|
|
|
|
d.addCallback(_got)
|
|
|
|
d.addCallback(lambda res: entries)
|
2007-06-15 07:37:32 +00:00
|
|
|
return d
|
|
|
|
|
2007-06-25 20:23:51 +00:00
|
|
|
def _hash_name(self, name):
|
|
|
|
return hashutil.dir_name_hash(self._readkey, name)
|
|
|
|
|
2007-06-15 07:37:32 +00:00
|
|
|
def get(self, name):
|
2007-06-25 20:23:51 +00:00
|
|
|
H_name = self._hash_name(name)
|
|
|
|
d = self._rref.callRemote("get", self._index, H_name)
|
|
|
|
def _check_index_error(f):
|
|
|
|
f.trap(IndexError)
|
|
|
|
raise IndexError("get(index=%s): unable to find child named '%s'"
|
|
|
|
% (idlib.b2a(self._index), name))
|
|
|
|
d.addErrback(_check_index_error)
|
|
|
|
d.addCallback(lambda (E_write, E_read):
|
|
|
|
self._decrypt_child(E_write, E_read))
|
|
|
|
d.addCallback(self._create_node)
|
2007-06-15 07:37:32 +00:00
|
|
|
return d
|
|
|
|
|
2007-06-25 20:23:51 +00:00
|
|
|
def _set(self, name, write_child, read_child):
|
|
|
|
if not self._mutable:
|
|
|
|
return defer.fail(NotMutableError())
|
|
|
|
H_name = self._hash_name(name)
|
|
|
|
E_name = self._encrypt(self._readkey, name)
|
|
|
|
E_write = ""
|
|
|
|
if self._writekey and write_child:
|
|
|
|
E_write = self._encrypt(self._writekey, write_child)
|
|
|
|
E_read = self._encrypt(self._readkey, read_child)
|
|
|
|
d = self._rref.callRemote("set", self._index, self._write_enabler,
|
|
|
|
H_name, E_name, E_write, E_read)
|
2007-06-15 07:37:32 +00:00
|
|
|
return d
|
|
|
|
|
2007-06-25 20:23:51 +00:00
|
|
|
def set_uri(self, name, child_uri):
|
|
|
|
write, read = self._split_uri(child_uri)
|
|
|
|
return self._set(name, write, read)
|
|
|
|
|
|
|
|
def set_node(self, name, child):
|
|
|
|
d = self.set_uri(name, child.get_uri())
|
|
|
|
d.addCallback(lambda res: child)
|
2007-06-15 07:37:32 +00:00
|
|
|
return d
|
|
|
|
|
2007-06-25 20:23:51 +00:00
|
|
|
def delete(self, name):
|
|
|
|
if not self._mutable:
|
|
|
|
return defer.fail(NotMutableError())
|
|
|
|
H_name = self._hash_name(name)
|
|
|
|
d = self._rref.callRemote("delete", self._index, self._write_enabler,
|
|
|
|
H_name)
|
2007-06-15 07:37:32 +00:00
|
|
|
return d
|
|
|
|
|
2007-06-25 20:23:51 +00:00
|
|
|
def _create_node(self, child_uri):
|
|
|
|
if uri.is_dirnode_uri(child_uri):
|
|
|
|
return create_directory_node(self._client, child_uri)
|
|
|
|
else:
|
|
|
|
return defer.succeed(FileNode(child_uri, self._client))
|
|
|
|
|
|
|
|
def _split_uri(self, child_uri):
|
|
|
|
if uri.is_dirnode_uri(child_uri):
|
|
|
|
if uri.is_mutable_dirnode_uri(child_uri):
|
|
|
|
write = child_uri
|
|
|
|
read = uri.make_immutable_dirnode_uri(child_uri)
|
|
|
|
else:
|
|
|
|
write = None
|
|
|
|
read = child_uri
|
|
|
|
return (write, read)
|
|
|
|
return (None, child_uri) # file
|
|
|
|
|
2007-06-15 07:37:32 +00:00
|
|
|
def create_empty_directory(self, name):
|
2007-06-25 20:23:51 +00:00
|
|
|
if not self._mutable:
|
|
|
|
return defer.fail(NotMutableError())
|
|
|
|
child_writekey = hashutil.random_key()
|
|
|
|
my_furl, parent_writekey = uri.unpack_dirnode_uri(self._uri)
|
|
|
|
child_uri = uri.pack_dirnode_uri(my_furl, child_writekey)
|
|
|
|
child = MutableDirectoryNode(child_uri, self._client, self._rref,
|
|
|
|
child_writekey)
|
|
|
|
d = self._rref.callRemote("create_directory",
|
|
|
|
child._index, child._write_enabler)
|
|
|
|
d.addCallback(lambda index: self.set_node(name, child))
|
2007-06-15 07:37:32 +00:00
|
|
|
return d
|
|
|
|
|
2007-06-25 20:23:51 +00:00
|
|
|
def add_file(self, name, uploadable):
|
|
|
|
if not self._mutable:
|
|
|
|
return defer.fail(NotMutableError())
|
|
|
|
uploader = self._client.getServiceNamed("uploader")
|
|
|
|
d = uploader.upload(uploadable)
|
|
|
|
d.addCallback(lambda uri: self.set_node(name,
|
|
|
|
FileNode(uri, self._client)))
|
2007-06-15 07:37:32 +00:00
|
|
|
return d
|
|
|
|
|
|
|
|
def move_child_to(self, current_child_name,
|
|
|
|
new_parent, new_child_name=None):
|
2007-06-25 20:23:51 +00:00
|
|
|
if not (self._mutable and new_parent.is_mutable()):
|
|
|
|
return defer.fail(NotMutableError())
|
2007-06-15 07:37:32 +00:00
|
|
|
if new_child_name is None:
|
|
|
|
new_child_name = current_child_name
|
|
|
|
d = self.get(current_child_name)
|
2007-06-25 20:23:51 +00:00
|
|
|
d.addCallback(lambda child: new_parent.set_node(new_child_name, child))
|
|
|
|
d.addCallback(lambda child: self.delete(current_child_name))
|
2007-06-15 07:37:32 +00:00
|
|
|
return d
|
|
|
|
|
2007-06-25 20:23:51 +00:00
|
|
|
class MutableDirectoryNode(ImmutableDirectoryNode):
|
|
|
|
implements(IDirectoryNode)
|
|
|
|
|
|
|
|
def __init__(self, myuri, client, rref, writekey):
|
|
|
|
readkey = hashutil.dir_read_key_hash(writekey)
|
|
|
|
ImmutableDirectoryNode.__init__(self, myuri, client, rref, readkey)
|
|
|
|
self._writekey = writekey
|
|
|
|
self._write_enabler = hashutil.dir_write_enabler_hash(writekey)
|
|
|
|
self._mutable = True
|
|
|
|
|
|
|
|
def create_directory(client, furl):
|
|
|
|
write_key = hashutil.random_key()
|
|
|
|
(wk, we, rk, index) = \
|
|
|
|
hashutil.generate_dirnode_keys_from_writekey(write_key)
|
|
|
|
myuri = uri.pack_dirnode_uri(furl, wk)
|
|
|
|
d = client.tub.getReference(furl)
|
|
|
|
def _got_vdrive_server(vdrive_server):
|
|
|
|
node = MutableDirectoryNode(myuri, client, vdrive_server, wk)
|
|
|
|
d2 = vdrive_server.callRemote("create_directory", index, we)
|
|
|
|
d2.addCallback(lambda res: node)
|
|
|
|
return d2
|
|
|
|
d.addCallback(_got_vdrive_server)
|
|
|
|
return d
|
|
|
|
|
|
|
|
class FileNode:
|
|
|
|
implements(IFileNode)
|
|
|
|
|
|
|
|
def __init__(self, uri, client):
|
2007-06-15 07:37:32 +00:00
|
|
|
self.uri = uri
|
|
|
|
self._client = client
|
2007-06-25 20:23:51 +00:00
|
|
|
|
|
|
|
def get_uri(self):
|
|
|
|
return self.uri
|
|
|
|
|
2007-06-15 07:37:32 +00:00
|
|
|
def __hash__(self):
|
|
|
|
return hash((self.__class__, self.uri))
|
|
|
|
def __cmp__(self, them):
|
|
|
|
if cmp(type(self), type(them)):
|
|
|
|
return cmp(type(self), type(them))
|
|
|
|
if cmp(self.__class__, them.__class__):
|
|
|
|
return cmp(self.__class__, them.__class__)
|
|
|
|
return cmp(self.uri, them.uri)
|
|
|
|
|
|
|
|
def download(self, target):
|
|
|
|
downloader = self._client.getServiceNamed("downloader")
|
|
|
|
return downloader.download(self.uri, target)
|
2007-06-15 03:14:34 +00:00
|
|
|
|
2007-06-15 07:37:32 +00:00
|
|
|
def download_to_data(self):
|
|
|
|
downloader = self._client.getServiceNamed("downloader")
|
|
|
|
return downloader.download_to_data(self.uri)
|
2007-06-15 03:14:34 +00:00
|
|
|
|