2007-06-27 17:11:06 -07:00
|
|
|
|
|
|
|
import os
|
|
|
|
from twisted.application import service
|
|
|
|
from zope.interface import implements
|
2007-07-21 15:40:36 -07:00
|
|
|
from allmydata.interfaces import IVirtualDrive, IDirnodeURI, IURI
|
2007-08-22 14:54:34 -07:00
|
|
|
from allmydata.util import observer
|
2007-07-21 15:40:36 -07:00
|
|
|
from allmydata import dirnode
|
2007-06-27 17:11:06 -07:00
|
|
|
from twisted.internet import defer
|
|
|
|
|
|
|
|
class NoGlobalVirtualDriveError(Exception):
|
|
|
|
pass
|
|
|
|
class NoPrivateVirtualDriveError(Exception):
|
|
|
|
pass
|
|
|
|
|
|
|
|
class VirtualDrive(service.MultiService):
|
|
|
|
implements(IVirtualDrive)
|
|
|
|
name = "vdrive"
|
|
|
|
|
|
|
|
GLOBAL_VDRIVE_FURL_FILE = "vdrive.furl"
|
|
|
|
|
|
|
|
GLOBAL_VDRIVE_URI_FILE = "global_root.uri"
|
|
|
|
MY_VDRIVE_URI_FILE = "my_vdrive.uri"
|
|
|
|
|
|
|
|
def __init__(self):
|
|
|
|
service.MultiService.__init__(self)
|
|
|
|
self._global_uri = None
|
|
|
|
self._private_uri = None
|
2007-08-22 14:54:34 -07:00
|
|
|
self._private_root_observer = observer.OneShotObserverList()
|
2007-06-27 17:11:06 -07:00
|
|
|
|
2007-07-24 19:44:25 -07:00
|
|
|
def log(self, msg):
|
|
|
|
self.parent.log(msg)
|
|
|
|
|
2007-06-27 17:11:06 -07:00
|
|
|
def startService(self):
|
|
|
|
service.MultiService.startService(self)
|
|
|
|
basedir = self.parent.basedir
|
|
|
|
tub = self.parent.tub
|
|
|
|
|
2007-07-24 19:44:25 -07:00
|
|
|
global_uri_file = os.path.join(basedir, self.GLOBAL_VDRIVE_URI_FILE)
|
2007-06-27 17:11:06 -07:00
|
|
|
if os.path.exists(global_uri_file):
|
|
|
|
f = open(global_uri_file)
|
|
|
|
self._global_uri = f.read().strip()
|
|
|
|
f.close()
|
2007-07-24 19:44:25 -07:00
|
|
|
self.log("using global vdrive uri %s" % self._global_uri)
|
2007-06-27 17:11:06 -07:00
|
|
|
|
2007-07-24 19:44:25 -07:00
|
|
|
private_uri_file = os.path.join(basedir, self.MY_VDRIVE_URI_FILE)
|
2007-06-27 17:11:06 -07:00
|
|
|
if os.path.exists(private_uri_file):
|
|
|
|
f = open(private_uri_file)
|
2007-06-28 10:59:51 -07:00
|
|
|
self._private_uri = f.read().strip()
|
2007-06-27 17:11:06 -07:00
|
|
|
f.close()
|
2007-07-24 19:44:25 -07:00
|
|
|
self.log("using private vdrive uri %s" % self._private_uri)
|
2007-08-22 14:54:34 -07:00
|
|
|
self._private_root_observer.fire(self._private_uri)
|
2007-07-24 19:44:25 -07:00
|
|
|
|
|
|
|
furl_file = os.path.join(basedir, self.GLOBAL_VDRIVE_FURL_FILE)
|
|
|
|
if os.path.exists(furl_file):
|
|
|
|
f = open(furl_file, "r")
|
|
|
|
global_vdrive_furl = f.read().strip()
|
|
|
|
f.close()
|
|
|
|
else:
|
|
|
|
self.log("no %s, cannot access global or private dirnodes"
|
|
|
|
% furl_file)
|
|
|
|
return
|
|
|
|
|
|
|
|
tub.connectTo(global_vdrive_furl,
|
|
|
|
self._got_vdrive_server, global_vdrive_furl)
|
|
|
|
|
|
|
|
if not self._global_uri:
|
|
|
|
self.log("will fetch global_uri when we attach to the "
|
|
|
|
"vdrive server")
|
|
|
|
|
|
|
|
if not self._private_uri:
|
|
|
|
self.log("will create private_uri when we attach to the "
|
|
|
|
"vdrive server")
|
|
|
|
|
|
|
|
|
|
|
|
def _got_vdrive_server(self, vdrive_server, global_vdrive_furl):
|
|
|
|
self.log("connected to vdrive server")
|
|
|
|
basedir = self.parent.basedir
|
|
|
|
global_uri_file = os.path.join(basedir, self.GLOBAL_VDRIVE_URI_FILE)
|
|
|
|
private_uri_file = os.path.join(basedir, self.MY_VDRIVE_URI_FILE)
|
|
|
|
|
|
|
|
d = vdrive_server.callRemote("get_public_root_uri")
|
|
|
|
def _got_global_uri(global_uri):
|
|
|
|
self.log("got global_uri: %s" % global_uri)
|
|
|
|
self._global_uri = global_uri
|
|
|
|
f = open(global_uri_file, "w")
|
|
|
|
f.write(self._global_uri + "\n")
|
|
|
|
f.close()
|
|
|
|
d.addCallback(_got_global_uri)
|
|
|
|
|
|
|
|
if not self._private_uri:
|
|
|
|
d.addCallback(lambda res:
|
|
|
|
dirnode.create_directory(self.parent,
|
|
|
|
global_vdrive_furl))
|
2007-06-27 17:11:06 -07:00
|
|
|
def _got_directory(dirnode):
|
2007-07-24 19:44:25 -07:00
|
|
|
self.log("creating private uri")
|
2007-06-28 10:59:51 -07:00
|
|
|
self._private_uri = dirnode.get_uri()
|
2007-06-27 17:11:06 -07:00
|
|
|
f = open(private_uri_file, "w")
|
2007-06-28 10:59:51 -07:00
|
|
|
f.write(self._private_uri + "\n")
|
2007-06-27 17:11:06 -07:00
|
|
|
f.close()
|
2007-08-22 14:54:34 -07:00
|
|
|
self._private_root_observer.fire(self._private_uri)
|
2007-06-27 17:11:06 -07:00
|
|
|
d.addCallback(_got_directory)
|
|
|
|
|
|
|
|
|
|
|
|
def have_public_root(self):
|
|
|
|
return bool(self._global_uri)
|
|
|
|
def get_public_root(self):
|
|
|
|
if not self._global_uri:
|
|
|
|
return defer.fail(NoGlobalVirtualDriveError())
|
|
|
|
return self.get_node(self._global_uri)
|
|
|
|
|
2007-08-22 14:54:34 -07:00
|
|
|
def when_private_root_available(self):
|
|
|
|
"""Return a Deferred that will fire with the URI of the private
|
|
|
|
vdrive root, when it is available.
|
|
|
|
|
|
|
|
This might be right away if the private vdrive was already present.
|
|
|
|
The first time the node is started, this will take a bit longer.
|
|
|
|
"""
|
|
|
|
return self._private_root_observer.when_fired()
|
|
|
|
|
2007-06-27 17:11:06 -07:00
|
|
|
def have_private_root(self):
|
|
|
|
return bool(self._private_uri)
|
|
|
|
def get_private_root(self):
|
|
|
|
if not self._private_uri:
|
|
|
|
return defer.fail(NoPrivateVirtualDriveError())
|
|
|
|
return self.get_node(self._private_uri)
|
|
|
|
|
2007-06-28 11:00:03 -07:00
|
|
|
def get_node(self, node_uri):
|
2007-07-21 15:40:36 -07:00
|
|
|
node_uri = IURI(node_uri)
|
|
|
|
if IDirnodeURI.providedBy(node_uri):
|
2007-06-28 11:00:03 -07:00
|
|
|
return dirnode.create_directory_node(self.parent, node_uri)
|
|
|
|
else:
|
|
|
|
return defer.succeed(dirnode.FileNode(node_uri, self.parent))
|
|
|
|
|
|
|
|
|
|
|
|
def get_node_at_path(self, path, root=None):
|
2007-06-28 17:46:14 -07:00
|
|
|
if not isinstance(path, (list, tuple)):
|
|
|
|
assert isinstance(path, (str, unicode))
|
|
|
|
if path[0] == "/":
|
|
|
|
path = path[1:]
|
|
|
|
path = path.split("/")
|
2007-06-28 11:00:03 -07:00
|
|
|
assert isinstance(path, (list, tuple))
|
|
|
|
|
|
|
|
if root is None:
|
|
|
|
if path and path[0] == "~":
|
|
|
|
d = self.get_private_root()
|
|
|
|
d.addCallback(lambda node:
|
|
|
|
self.get_node_at_path(path[1:], node))
|
|
|
|
return d
|
|
|
|
d = self.get_public_root()
|
|
|
|
d.addCallback(lambda node: self.get_node_at_path(path, node))
|
|
|
|
return d
|
|
|
|
|
|
|
|
if path:
|
|
|
|
assert path[0] != ""
|
|
|
|
d = root.get(path[0])
|
|
|
|
d.addCallback(lambda node: self.get_node_at_path(path[1:], node))
|
|
|
|
return d
|
|
|
|
|
|
|
|
return root
|