From a312dd10e89ffc3397d16a6e4047ebc5f4c21870 Mon Sep 17 00:00:00 2001 From: Brian Warner <warner@lothar.com> Date: Mon, 4 Dec 2006 19:54:35 -0700 Subject: [PATCH] webish: add mesh stats, peer list. improve distribution of client services within rend_* methods --- allmydata/web/welcome.xhtml | 22 +++++++++++++++- allmydata/webish.py | 50 ++++++++++++++++++++++++++++++------- 2 files changed, 62 insertions(+), 10 deletions(-) diff --git a/allmydata/web/welcome.xhtml b/allmydata/web/welcome.xhtml index 8ff2a8ba..9c0fda17 100644 --- a/allmydata/web/welcome.xhtml +++ b/allmydata/web/welcome.xhtml @@ -7,9 +7,29 @@ </head> <body> -<h1>Welcome To AllMyData!</h1> +<h1>Welcome To AllMyData! (tahoe2)</h1> <p>To view the global shared filestore, <a href="vdrive">Click Here!</a></p> +<h2>Mesh Status</h2> + +<div>Known Peers: <span n:render="string" n:data="num_peers" /></div> +<div>Connected Peers: <span n:render="string" n:data="num_connected_peers" /></div> + +<div> +<table n:render="sequence" n:data="peers" border="1"> + <tr n:pattern="header"> + <td>PeerID</td> + <td>Connected?</td> + </tr> + <tr n:pattern="item" n:render="row"> + <td><tt><n:slot name="peerid"/></tt></td> + <td><n:slot name="connected"/></td> + </tr> + <tr n:pattern="empty"><td>no peers!</td></tr> +</table> +</div> + + </body> </html> diff --git a/allmydata/webish.py b/allmydata/webish.py index 4d41cbc0..83eb8b15 100644 --- a/allmydata/webish.py +++ b/allmydata/webish.py @@ -6,13 +6,16 @@ from nevow import inevow, rend, loaders, appserver, url, tags as T from allmydata.util import idlib from allmydata.download import IDownloadTarget#, IDownloader from allmydata import upload -from zope.interface import implements +from zope.interface import implements, Interface import urllib from formless import annotate, webform def getxmlfile(name): return loaders.xmlfile(util.sibpath(__file__, "web/%s" % name)) +class IClient(Interface): + pass + class WebishServer(service.MultiService): name = "webish" WEBPORTFILE = "webport" @@ -27,8 +30,20 @@ class WebishServer(service.MultiService): self.site = site = appserver.NevowSite(root) internet.TCPServer(webport, site).setServiceParent(self) + def startService(self): + service.MultiService.startService(self) + # to make various services available to render_* methods, we stash a + # reference to the client on the NevowSite. This will be available by + # adapting the 'context' argument to a special marker interface named + # IClient. + self.site.remember(self.parent, IClient) + # I thought you could do the same with an existing interface, but + # apparently 'ISite' does not exist + #self.site._client = self.parent + def set_root_dirnode(self, dirnode): - self.root.putChild("vdrive", Directory(dirnode, "/", self.parent)) + self.root.putChild("vdrive", Directory(dirnode, "/")) + # I tried doing it this way and for some reason it didn't seem to work #print "REMEMBERING", self.site, dl, IDownloader #self.site.remember(dl, IDownloader) @@ -37,15 +52,33 @@ class Welcome(rend.Page): addSlash = True docFactory = getxmlfile("welcome.xhtml") + def data_num_peers(self, ctx, data): + #client = inevow.ISite(ctx)._client + client = IClient(ctx) + return len(client.all_peers) + def data_num_connected_peers(self, ctx, data): + return len(IClient(ctx).connections) + def data_peers(self, ctx, data): + return sorted(IClient(ctx).all_peers) + def render_row(self, ctx, data): + if data in IClient(ctx).connections: + connected = "yes" + else: + connected = "no" + ctx.fillSlots("peerid", idlib.b2a(data)) + ctx.fillSlots("connected", connected) + return ctx.tag class Directory(rend.Page): addSlash = True docFactory = getxmlfile("directory.xhtml") - def __init__(self, dirnode, dirname, client): + def __init__(self, dirnode, dirname): self._dirnode = dirnode self._dirname = dirname - self._client = client + + def get_service(self, ctx, name): + return IClient(ctx).getServiceNamed(name) def childFactory(self, ctx, name): if name.startswith("freeform"): # ick @@ -54,15 +87,14 @@ class Directory(rend.Page): args = inevow.IRequest(ctx).args filename = args["filename"][0] verifierid = args["verifierid"][0] - return Downloader(self._client.getServiceNamed("downloader"), + return Downloader(self.get_service(ctx, "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._client)) + d.addCallback(lambda newnode: Directory(newnode, dirname)) return d def render_title(self, ctx, data): @@ -135,7 +167,7 @@ class Directory(rend.Page): # contents is a cgi.FieldStorage instance log.msg("starting webish upload") - uploader = self._client.getServiceNamed("uploader") + uploader = self.get_service(ctx, "uploader") d = uploader.upload(upload.Data(contents.value)) name = contents.filename d.addCallback(lambda vid: @@ -168,7 +200,7 @@ class Directory(rend.Page): def child__delete(self, ctx): # perform the delete, then redirect back to the directory page args = inevow.IRequest(ctx).args - vdrive = self._client.getServiceNamed("vdrive") + vdrive = self.get_service(ctx, "vdrive") d = vdrive.remove(self._dirnode, args["name"][0]) def _deleted(res): return url.here.up() -- 2.45.2