From a3700cc582342d013efbec2d63a03e68e54ac1f9 Mon Sep 17 00:00:00 2001
From: Brian Warner <warner@lothar.com>
Date: Mon, 4 Dec 2006 04:06:09 -0700
Subject: [PATCH] add a (read-only) web frontend. Put a 'webport' file in your
 base directory to activate it.

---
 allmydata/client.py           |  10 +++
 allmydata/download.py         |   4 +
 allmydata/web/directory.xhtml |  36 +++++++++
 allmydata/web/welcome.xhtml   |  15 ++++
 allmydata/webish.py           | 145 ++++++++++++++++++++++++++++++++++
 roadmap.txt                   |   3 +
 6 files changed, 213 insertions(+)
 create mode 100644 allmydata/web/directory.xhtml
 create mode 100644 allmydata/web/welcome.xhtml
 create mode 100644 allmydata/webish.py

diff --git a/allmydata/client.py b/allmydata/client.py
index 8afc3ac2..f0ecd6e6 100644
--- a/allmydata/client.py
+++ b/allmydata/client.py
@@ -14,6 +14,7 @@ from allmydata.storageserver import StorageServer
 from allmydata.upload import Uploader
 from allmydata.download import Downloader
 from allmydata.vdrive import VDrive
+from allmydata.webish import WebishServer
 
 class Client(node.Node, Referenceable):
     implements(RIClient)
@@ -21,6 +22,7 @@ class Client(node.Node, Referenceable):
     PORTNUMFILE = "client.port"
     STOREDIR = 'storage'
     NODETYPE = "client"
+    WEBPORTFILE = "webport"
 
     def __init__(self, basedir="."):
         node.Node.__init__(self, basedir)
@@ -31,6 +33,12 @@ class Client(node.Node, Referenceable):
         self.add_service(Uploader())
         self.add_service(Downloader())
         self.add_service(VDrive())
+        WEBPORTFILE = os.path.join(self.basedir, self.WEBPORTFILE)
+        if os.path.exists(WEBPORTFILE):
+            f = open(WEBPORTFILE, "r")
+            webport = int(f.read())
+            f.close()
+            self.add_service(WebishServer(webport))
         self.queen_pburl = None
         self.queen_connector = None
 
@@ -73,6 +81,8 @@ class Client(node.Node, Referenceable):
 
     def _got_vdrive_root(self, root):
         self.getServiceNamed("vdrive").set_root(root)
+        if "webish" in self.namedServices:
+            self.getServiceNamed("webish").set_root_dirnode(root)
 
     def _lost_queen(self):
         self.log("lost connection to queen")
diff --git a/allmydata/download.py b/allmydata/download.py
index 58b8b4b2..a344d039 100644
--- a/allmydata/download.py
+++ b/allmydata/download.py
@@ -195,10 +195,14 @@ class FileHandle:
     def finish(self):
         pass
 
+class IDownloader(Interface):
+    def download(verifierid, target):
+        pass
 
 class Downloader(service.MultiService):
     """I am a service that allows file downloading.
     """
+    implements(IDownloader)
     name = "downloader"
 
     def download(self, verifierid, t):
diff --git a/allmydata/web/directory.xhtml b/allmydata/web/directory.xhtml
new file mode 100644
index 00000000..1e68ca3d
--- /dev/null
+++ b/allmydata/web/directory.xhtml
@@ -0,0 +1,36 @@
+<html xmlns:n="http://nevow.com/ns/nevow/0.1">
+  <head>
+    <title n:render="title"></title>
+    <link href="http://www.allmydata.com/common/css/styles.css"
+          rel="stylesheet" type="text/css"/>
+    <meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
+  </head>
+  <body>
+
+<h1><p n:render="header"></p></h1>
+
+<div><a href=".">Refresh this view</a></div>
+<div><a href="..">Parent Directory</a></div>
+
+<table n:render="sequence" n:data="children" border="1">
+  <tr n:pattern="header">
+    <td>Filename</td>
+    <td>Type</td>
+    <td>fileid</td>
+  </tr>
+  <tr n:pattern="item" n:render="row">
+    <td><n:slot name="filename"/></td>
+    <td><n:slot name="type"/></td>
+    <td><n:slot name="fileid"/></td>
+  </tr>
+
+  <tr n:pattern="empty"><td>directory is empty!</td></tr>
+
+</table>
+
+<!-- <div n:render="forms"/> -->
+
+<!-- <div class="results" n:render="results"/> -->
+
+  </body>
+</html>
diff --git a/allmydata/web/welcome.xhtml b/allmydata/web/welcome.xhtml
new file mode 100644
index 00000000..8ff2a8ba
--- /dev/null
+++ b/allmydata/web/welcome.xhtml
@@ -0,0 +1,15 @@
+<html xmlns:n="http://nevow.com/ns/nevow/0.1">
+  <head>
+    <title>Welcome To AllMyData (tahoe2)</title>
+    <link href="http://www.allmydata.com/common/css/styles.css"
+          rel="stylesheet" type="text/css"/>
+    <meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
+  </head>
+  <body>
+
+<h1>Welcome To AllMyData!</h1>
+
+<p>To view the global shared filestore, <a href="vdrive">Click Here!</a></p>
+
+  </body>
+</html>
diff --git a/allmydata/webish.py b/allmydata/webish.py
new file mode 100644
index 00000000..499f4051
--- /dev/null
+++ b/allmydata/webish.py
@@ -0,0 +1,145 @@
+
+from twisted.application import service, internet
+from twisted.web import static, resource, server
+from twisted.python import util
+from nevow import inevow, rend, loaders, appserver, tags as T
+from allmydata.util import idlib
+from allmydata.download import IDownloadTarget#, IDownloader
+from zope.interface import implements
+import urllib
+
+def getxmlfile(name):
+    return loaders.xmlfile(util.sibpath(__file__, "web/%s" % name))
+
+class WebishServer(service.MultiService):
+    name = "webish"
+    WEBPORTFILE = "webport"
+
+    def __init__(self, webport):
+        service.MultiService.__init__(self)
+        self.root = root = static.Data("root", "text/plain")
+        w = Welcome()
+        root.putChild("", w)
+        root.putChild("vdrive",
+                      static.Data("sorry, still initializing", "text/plain"))
+        self.site = site = appserver.NevowSite(root)
+        internet.TCPServer(webport, site).setServiceParent(self)
+
+    def set_root_dirnode(self, dirnode):
+        dl = self.parent.getServiceNamed("downloader")
+        self.root.putChild("vdrive", Directory(dirnode, "/", dl))
+        #print "REMEMBERING", self.site, dl, IDownloader
+        #self.site.remember(dl, IDownloader)
+
+
+class Welcome(rend.Page):
+    addSlash = True
+    docFactory = getxmlfile("welcome.xhtml")
+
+class Directory(rend.Page):
+    addSlash = True
+    docFactory = getxmlfile("directory.xhtml")
+
+    def __init__(self, dirnode, dirname, downloader):
+        self._dirnode = dirnode
+        self._dirname = dirname
+        self._downloader = downloader
+
+    def childFactory(self, ctx, name):
+        if name.startswith("freeform"): # ick
+            return None
+        if name == "_download":
+            args = inevow.IRequest(ctx).args
+            filename = args["filename"][0]
+            verifierid = args["verifierid"][0]
+            return Downloader(self._downloader,
+                              self._dirname, filename, idlib.a2b(verifierid))
+        if self._dirname == "/":
+            dirname = "/" + name
+        else:
+            dirname = self._dirname + "/" + name
+        d = self._dirnode.callRemote("get", name)
+        d.addCallback(lambda newnode:
+                      Directory(newnode, dirname, self._downloader))
+        return d
+
+    def render_title(self, ctx, data):
+        return ctx.tag["Directory of '%s':" % self._dirname]
+
+    def render_header(self, ctx, data):
+        return "Directory of '%s':" % self._dirname
+
+    def data_children(self, ctx, data):
+        d = self._dirnode.callRemote("list")
+        return d
+
+    def render_row(self, ctx, data):
+        name, target = data
+        if isinstance(target, str):
+            # file
+            args = {'verifierid': idlib.b2a(target),
+                    'filename': name,
+                    }
+            dlurl = "_download?%s" % urllib.urlencode(args)
+            ctx.fillSlots("filename", T.a(href=dlurl)[name])
+            ctx.fillSlots("type", "FILE")
+            ctx.fillSlots("fileid", idlib.b2a(target))
+        else:
+            # directory
+            ctx.fillSlots("filename", T.a(href=name)[name])
+            ctx.fillSlots("type", "DIR")
+            ctx.fillSlots("fileid", "-")
+        return ctx.tag
+
+class WebDownloadTarget:
+    implements(IDownloadTarget)
+    def __init__(self, req):
+        self._req = req
+    def open(self):
+        pass
+    def write(self, data):
+        self._req.write(data)
+    def close(self):
+        self._req.finish()
+    def fail(self):
+        self._req.finish()
+    def register_canceller(self, cb):
+        pass
+    def finish(self):
+        pass
+
+class TypedFile(static.File):
+    # serve data from a named file, but using a Content-Type derived from a
+    # different filename
+    isLeaf = True
+    def __init__(self, path, requested_filename):
+        static.File.__init__(self, path)
+        gte = static.getTypeAndEncoding
+        self.type, self.encoding = gte(requested_filename,
+                                       self.contentTypes,
+                                       self.contentEncodings,
+                                       self.defaultType)
+
+class Downloader(resource.Resource):
+    def __init__(self, downloader, dirname, name, verifierid):
+        self._downloader = downloader
+        self._dirname = dirname
+        self._name = name
+        self._verifierid = verifierid
+
+    def render(self, ctx):
+        req = inevow.IRequest(ctx)
+        gte = static.getTypeAndEncoding
+        type, encoding = gte(self._name,
+                             static.File.contentTypes,
+                             static.File.contentEncodings,
+                             defaultType="text/plain")
+        req.setHeader("content-type", type)
+        if encoding:
+            req.setHeader('content-encoding', encoding)
+
+        t = WebDownloadTarget(req)
+        #dl = IDownloader(ctx)
+        dl = self._downloader
+        dl.download(self._verifierid, t)
+        return server.NOT_DONE_YET
diff --git a/roadmap.txt b/roadmap.txt
index 14ee9d64..0f595eaf 100644
--- a/roadmap.txt
+++ b/roadmap.txt
@@ -28,6 +28,9 @@ storage: RobK
      leases never expire
  v2: leases expire, delete expired data on demand, multiple owners per share
 
+UI:
+ webish? webfront? PB + CLI tool? FUSE?
+ v1:
 
 back pocket ideas:
  when nodes are unable to reach storage servers, make a note of it, inform
-- 
2.45.2