2009-02-20 14:29:26 -07:00
|
|
|
|
|
|
|
from nevow import rend, tags as T
|
2009-02-20 21:04:08 -07:00
|
|
|
from allmydata.web.common import getxmlfile, abbreviate_time
|
|
|
|
from allmydata.util.abbreviate import abbreviate_space
|
2009-02-20 14:29:26 -07:00
|
|
|
|
|
|
|
def remove_prefix(s, prefix):
|
|
|
|
if not s.startswith(prefix):
|
|
|
|
return None
|
|
|
|
return s[len(prefix):]
|
|
|
|
|
|
|
|
class StorageStatus(rend.Page):
|
|
|
|
docFactory = getxmlfile("storage_status.xhtml")
|
|
|
|
# the default 'data' argument is the StorageServer instance
|
|
|
|
|
|
|
|
def __init__(self, storage):
|
|
|
|
rend.Page.__init__(self, storage)
|
|
|
|
self.storage = storage
|
|
|
|
|
|
|
|
def render_storage_running(self, ctx, storage):
|
|
|
|
if storage:
|
|
|
|
return ctx.tag
|
|
|
|
else:
|
|
|
|
return T.h1["No Storage Server Running"]
|
|
|
|
|
|
|
|
def render_bool(self, ctx, data):
|
|
|
|
return {True: "Yes", False: "No"}[bool(data)]
|
|
|
|
|
2009-02-20 21:04:08 -07:00
|
|
|
def render_space(self, ctx, size):
|
|
|
|
if size is None:
|
|
|
|
return "?"
|
|
|
|
return "%s (%d)" % (abbreviate_space(size), size)
|
2009-02-20 14:29:26 -07:00
|
|
|
|
|
|
|
def data_stats(self, ctx, data):
|
|
|
|
# FYI: 'data' appears to be self, rather than the StorageServer
|
|
|
|
# object in self.original that gets passed to render_* methods. I
|
|
|
|
# still don't understand Nevow.
|
|
|
|
|
2009-02-20 16:03:53 -07:00
|
|
|
# Nevow has nevow.accessors.DictionaryContainer: Any data= directive
|
|
|
|
# that appears in a context in which the current data is a dictionary
|
|
|
|
# will be looked up as keys in that dictionary. So if data_stats()
|
|
|
|
# returns a dictionary, then we can use something like this:
|
|
|
|
#
|
2009-02-20 14:29:26 -07:00
|
|
|
# <ul n:data="stats">
|
2009-02-20 16:03:53 -07:00
|
|
|
# <li>disk_total: <span n:render="abbrev" n:data="disk_total" /></li>
|
2009-02-20 14:29:26 -07:00
|
|
|
# </ul>
|
2009-02-20 16:03:53 -07:00
|
|
|
|
|
|
|
# to use get_stats()["storage_server.disk_total"] . However,
|
|
|
|
# DictionaryContainer does a raw d[] instead of d.get(), so any
|
|
|
|
# missing keys will cause an error, even if the renderer can tolerate
|
|
|
|
# None values. To overcome this, we either need a dict-like object
|
|
|
|
# that always returns None for unknown keys, or we must pre-populate
|
2009-02-20 21:04:08 -07:00
|
|
|
# our dict with those missing keys, or we should get rid of data_
|
|
|
|
# methods that return dicts (or find some way to override Nevow's
|
|
|
|
# handling of dictionaries).
|
2009-02-20 16:03:53 -07:00
|
|
|
|
|
|
|
d = dict([ (remove_prefix(k, "storage_server."), v)
|
|
|
|
for k,v in self.storage.get_stats().items() ])
|
|
|
|
d.setdefault("disk_total", None)
|
|
|
|
d.setdefault("disk_used", None)
|
2009-02-20 21:28:56 -07:00
|
|
|
d.setdefault("disk_free_for_root", None)
|
|
|
|
d.setdefault("disk_free_for_nonroot", None)
|
2009-02-20 16:03:53 -07:00
|
|
|
d.setdefault("reserved_space", None)
|
|
|
|
d.setdefault("disk_avail", None)
|
|
|
|
return d
|
2009-02-20 21:04:08 -07:00
|
|
|
|
2009-02-20 21:46:06 -07:00
|
|
|
def data_last_complete_bucket_count(self, ctx, data):
|
2009-02-20 21:04:08 -07:00
|
|
|
s = self.storage.bucket_counter.get_state()
|
2009-02-20 21:46:06 -07:00
|
|
|
lcbc = s.get("last-complete-bucket-count")
|
|
|
|
if lcbc is None:
|
2009-02-20 21:04:08 -07:00
|
|
|
return "Not computed yet"
|
2009-02-20 21:46:06 -07:00
|
|
|
cycle, count = lcbc
|
2009-02-20 21:04:08 -07:00
|
|
|
return count
|
|
|
|
|
|
|
|
def render_count_crawler_status(self, ctx, storage):
|
|
|
|
s = self.storage.bucket_counter.get_progress()
|
|
|
|
if s["cycle-in-progress"]:
|
|
|
|
pct = s["cycle-complete-percentage"]
|
|
|
|
soon = s["remaining-sleep-time"]
|
|
|
|
return ctx.tag["Current crawl %.1f%% complete" % pct,
|
|
|
|
" (next work in %s)" % abbreviate_time(soon)]
|
|
|
|
else:
|
|
|
|
soon = s["remaining-wait-time"]
|
|
|
|
return ctx.tag["Next crawl in %s" % abbreviate_time(soon)]
|