2008-07-16 22:42:56 +00:00
2008-08-13 03:34:26 +00:00
import time
2008-09-10 02:45:17 +00:00
import simplejson
2008-07-16 22:42:56 +00:00
from nevow import rend , inevow , tags as T
2008-10-23 23:00:31 +00:00
from twisted . web import http , html
2009-02-20 19:15:54 +00:00
from allmydata . web . common import getxmlfile , get_arg , get_root , WebError
2008-10-22 00:03:07 +00:00
from allmydata . web . operations import ReloadMixin
2009-01-06 20:37:03 +00:00
from allmydata . interfaces import ICheckAndRepairResults , ICheckResults
2008-09-07 19:44:56 +00:00
from allmydata . util import base32 , idlib
2008-07-16 22:42:56 +00:00
2009-02-17 06:35:53 +00:00
def json_check_counts ( d ) :
r = { }
r [ " count-shares-good " ] = d [ " count-shares-good " ]
r [ " count-shares-needed " ] = d [ " count-shares-needed " ]
r [ " count-shares-expected " ] = d [ " count-shares-expected " ]
r [ " count-good-share-hosts " ] = d [ " count-good-share-hosts " ]
r [ " count-corrupt-shares " ] = d [ " count-corrupt-shares " ]
r [ " list-corrupt-shares " ] = [ ( idlib . nodeid_b2a ( serverid ) ,
base32 . b2a ( si ) , shnum )
for ( serverid , si , shnum )
in d [ " list-corrupt-shares " ] ]
r [ " servers-responding " ] = [ idlib . nodeid_b2a ( serverid )
for serverid in d [ " servers-responding " ] ]
sharemap = { }
for ( shareid , serverids ) in d [ " sharemap " ] . items ( ) :
sharemap [ shareid ] = [ idlib . nodeid_b2a ( serverid )
for serverid in serverids ]
r [ " sharemap " ] = sharemap
r [ " count-wrong-shares " ] = d [ " count-wrong-shares " ]
r [ " count-recoverable-versions " ] = d [ " count-recoverable-versions " ]
r [ " count-unrecoverable-versions " ] = d [ " count-unrecoverable-versions " ]
return r
def json_check_results ( r ) :
if r is None :
# LIT file
data = { " storage-index " : " " ,
" results " : { " healthy " : True } ,
}
return data
data = { }
data [ " storage-index " ] = r . get_storage_index_string ( )
data [ " summary " ] = r . get_summary ( )
data [ " results " ] = json_check_counts ( r . get_data ( ) )
data [ " results " ] [ " needs-rebalancing " ] = r . needs_rebalancing ( )
data [ " results " ] [ " healthy " ] = r . is_healthy ( )
data [ " results " ] [ " recoverable " ] = r . is_recoverable ( )
return data
def json_check_and_repair_results ( r ) :
if r is None :
# LIT file
data = { " storage-index " : " " ,
" repair-attempted " : False ,
}
return data
data = { }
data [ " storage-index " ] = r . get_storage_index_string ( )
data [ " repair-attempted " ] = r . get_repair_attempted ( )
data [ " repair-successful " ] = r . get_repair_successful ( )
pre = r . get_pre_repair_results ( )
data [ " pre-repair-results " ] = json_check_results ( pre )
post = r . get_post_repair_results ( )
data [ " post-repair-results " ] = json_check_results ( post )
return data
2008-09-07 19:44:56 +00:00
class ResultsBase :
2009-02-20 19:15:54 +00:00
# self.client must point to the Client, so we can get nicknames and
# determine the permuted peer order
2009-01-13 01:56:19 +00:00
def _join_pathstring ( self , path ) :
if path :
pathstring = " / " . join ( self . _html ( path ) )
else :
pathstring = " <root> "
return pathstring
2008-10-23 23:00:31 +00:00
def _render_results ( self , ctx , cr ) :
2009-01-06 20:37:03 +00:00
assert ICheckResults ( cr )
2009-02-20 19:15:54 +00:00
c = self . client
2009-06-02 03:07:50 +00:00
sb = c . get_storage_broker ( )
2008-10-23 23:00:31 +00:00
data = cr . get_data ( )
r = [ ]
def add ( name , value ) :
r . append ( T . li [ name + " : " , value ] )
add ( " Report " , T . pre [ " \n " . join ( self . _html ( cr . get_report ( ) ) ) ] )
add ( " Share Counts " ,
" need %d -of- %d , have %d " % ( data [ " count-shares-needed " ] ,
data [ " count-shares-expected " ] ,
data [ " count-shares-good " ] ) )
add ( " Hosts with good shares " , data [ " count-good-share-hosts " ] )
if data [ " list-corrupt-shares " ] :
badsharemap = [ ]
for ( serverid , si , shnum ) in data [ " list-corrupt-shares " ] :
2009-06-02 03:07:50 +00:00
nickname = sb . get_nickname_for_serverid ( serverid )
2008-10-23 23:00:31 +00:00
badsharemap . append ( T . tr [ T . td [ " sh# %d " % shnum ] ,
2009-05-26 23:25:45 +00:00
T . td [ T . div ( class_ = " nickname " ) [ nickname ] ,
T . div ( class_ = " nodeid " ) [ T . tt [ base32 . b2a ( serverid ) ] ] ] ,
2008-10-23 23:00:31 +00:00
] )
2009-05-26 23:25:45 +00:00
add ( " Corrupt shares " , T . table ( ) [
T . tr [ T . th [ " Share ID " ] ,
T . th ( class_ = " nickname-and-peerid " ) [ T . div [ " Nickname " ] , T . div ( class_ = " nodeid " ) [ " Node ID " ] ] ] ,
badsharemap ] )
2008-10-23 23:00:31 +00:00
else :
add ( " Corrupt shares " , " none " )
add ( " Wrong Shares " , data [ " count-wrong-shares " ] )
sharemap = [ ]
servers = { }
2008-12-05 09:03:09 +00:00
2009-05-03 20:34:42 +00:00
# FIXME: The two tables below contain nickname-and-nodeid table column markup which is duplicated with each other, introducer.xhtml, and deep-check-results.xhtml. All of these (and any other presentations of nickname-and-nodeid) should be combined.
2008-10-23 23:00:31 +00:00
for shareid in sorted ( data [ " sharemap " ] . keys ( ) ) :
serverids = data [ " sharemap " ] [ shareid ]
for i , serverid in enumerate ( serverids ) :
2008-12-05 09:03:09 +00:00
if serverid not in servers :
servers [ serverid ] = [ ]
servers [ serverid ] . append ( shareid )
2008-10-23 23:00:31 +00:00
shareid_s = " "
if i == 0 :
shareid_s = shareid
2009-06-02 03:07:50 +00:00
nickname = sb . get_nickname_for_serverid ( serverid )
2008-10-23 23:00:31 +00:00
sharemap . append ( T . tr [ T . td [ shareid_s ] ,
2009-05-26 23:25:45 +00:00
T . td [ T . div ( class_ = " nickname " ) [ nickname ] ,
T . div ( class_ = " nodeid " ) [ T . tt [ base32 . b2a ( serverid ) ] ] ]
2008-10-23 23:00:31 +00:00
] )
2008-12-05 09:03:09 +00:00
add ( " Good Shares (sorted in share order) " ,
2009-05-26 23:25:45 +00:00
T . table ( ) [ T . tr [ T . th [ " Share ID " ] , T . th ( class_ = " nickname-and-peerid " ) [ T . div [ " Nickname " ] , T . div ( class_ = " nodeid " ) [ " Node ID " ] ] ] ,
sharemap ] )
2008-12-05 09:03:09 +00:00
2008-10-23 23:00:31 +00:00
add ( " Recoverable Versions " , data [ " count-recoverable-versions " ] )
add ( " Unrecoverable Versions " , data [ " count-unrecoverable-versions " ] )
2008-12-05 09:03:09 +00:00
# this table is sorted by permuted order
2009-06-02 02:25:11 +00:00
sb = c . get_storage_broker ( )
2008-12-05 09:03:09 +00:00
permuted_peer_ids = [ peerid
for ( peerid , rref )
2009-06-21 23:51:19 +00:00
in sb . get_servers_for_index ( cr . get_storage_index ( ) ) ]
2008-12-05 09:03:09 +00:00
num_shares_left = sum ( [ len ( shares ) for shares in servers . values ( ) ] )
2008-10-23 23:00:31 +00:00
servermap = [ ]
2008-12-05 09:03:09 +00:00
for serverid in permuted_peer_ids :
2009-06-02 03:07:50 +00:00
nickname = sb . get_nickname_for_serverid ( serverid )
2008-12-05 09:03:09 +00:00
shareids = servers . get ( serverid , [ ] )
shareids . reverse ( )
2009-02-20 19:29:22 +00:00
shareids_s = [ T . tt [ shareid , " " ] for shareid in sorted ( shareids ) ]
2009-05-26 23:25:45 +00:00
servermap . append ( T . tr [ T . td [ T . div ( class_ = " nickname " ) [ nickname ] ,
T . div ( class_ = " nodeid " ) [ T . tt [ base32 . b2a ( serverid ) ] ] ] ,
T . td [ shareids_s ] ,
] )
2008-12-05 09:03:09 +00:00
num_shares_left - = len ( shareids )
if not num_shares_left :
break
add ( " Share Balancing (servers in permuted order) " ,
2009-05-26 23:25:45 +00:00
T . table ( ) [ T . tr [ T . th ( class_ = " nickname-and-peerid " ) [ T . div [ " Nickname " ] , T . div ( class_ = " nodeid " ) [ " Node ID " ] ] , T . th [ " Share IDs " ] ] ,
servermap ] )
2008-10-23 23:00:31 +00:00
return T . ul [ r ]
2008-09-10 02:45:17 +00:00
2008-09-07 19:44:56 +00:00
def _html ( self , s ) :
if isinstance ( s , ( str , unicode ) ) :
return html . escape ( s )
assert isinstance ( s , ( list , tuple ) )
return [ html . escape ( w ) for w in s ]
2008-09-10 21:11:37 +00:00
def want_json ( self , ctx ) :
output = get_arg ( inevow . IRequest ( ctx ) , " output " , " " ) . lower ( )
if output . lower ( ) == " json " :
return True
return False
2008-10-23 23:00:31 +00:00
def _render_si_link ( self , ctx , storage_index ) :
si_s = base32 . b2a ( storage_index )
req = inevow . IRequest ( ctx )
ophandle = req . prepath [ - 1 ]
target = " %s /operations/ %s / %s " % ( get_root ( ctx ) , ophandle , si_s )
output = get_arg ( ctx , " output " )
if output :
target = target + " ?output= %s " % output
return T . a ( href = target ) [ si_s ]
2009-01-10 01:00:52 +00:00
class LiteralCheckResults ( rend . Page , ResultsBase ) :
2009-01-06 20:37:03 +00:00
docFactory = getxmlfile ( " literal-check-results.xhtml " )
2008-09-10 02:45:17 +00:00
2009-02-20 19:15:54 +00:00
def __init__ ( self , client ) :
self . client = client
rend . Page . __init__ ( self , client )
2008-09-10 02:45:17 +00:00
def renderHTTP ( self , ctx ) :
2008-09-10 21:11:37 +00:00
if self . want_json ( ctx ) :
2008-09-10 02:45:17 +00:00
return self . json ( ctx )
return rend . Page . renderHTTP ( self , ctx )
def json ( self , ctx ) :
inevow . IRequest ( ctx ) . setHeader ( " content-type " , " text/plain " )
2009-02-23 21:19:43 +00:00
data = json_check_results ( None )
2008-09-15 20:43:14 +00:00
return simplejson . dumps ( data , indent = 1 ) + " \n "
2008-09-10 02:45:17 +00:00
2009-02-17 05:12:42 +00:00
def render_return ( self , ctx , data ) :
req = inevow . IRequest ( ctx )
return_to = get_arg ( req , " return_to " , None )
if return_to :
2009-04-07 18:54:59 +00:00
return T . div [ T . a ( href = return_to ) [ " Return to file. " ] ]
2009-02-17 05:12:42 +00:00
return " "
2008-10-23 23:00:31 +00:00
class CheckerBase :
2008-07-16 22:42:56 +00:00
2008-09-10 02:45:17 +00:00
def renderHTTP ( self , ctx ) :
2008-09-10 21:11:37 +00:00
if self . want_json ( ctx ) :
2008-09-10 02:45:17 +00:00
return self . json ( ctx )
return rend . Page . renderHTTP ( self , ctx )
2008-10-23 23:00:31 +00:00
def render_storage_index ( self , ctx , data ) :
return self . r . get_storage_index_string ( )
def render_return ( self , ctx , data ) :
req = inevow . IRequest ( ctx )
return_to = get_arg ( req , " return_to " , None )
if return_to :
2009-04-07 18:54:59 +00:00
return T . div [ T . a ( href = return_to ) [ " Return to file/directory. " ] ]
2008-10-23 23:00:31 +00:00
return " "
2009-01-10 01:00:52 +00:00
class CheckResults ( CheckerBase , rend . Page , ResultsBase ) :
2009-01-06 20:37:03 +00:00
docFactory = getxmlfile ( " check-results.xhtml " )
2008-10-23 23:00:31 +00:00
2009-02-20 19:15:54 +00:00
def __init__ ( self , client , results ) :
self . client = client
2009-01-06 20:37:03 +00:00
self . r = ICheckResults ( results )
2009-02-20 19:15:54 +00:00
rend . Page . __init__ ( self , results )
2008-10-23 23:00:31 +00:00
2008-09-10 02:45:17 +00:00
def json ( self , ctx ) :
inevow . IRequest ( ctx ) . setHeader ( " content-type " , " text/plain " )
2009-02-17 06:35:53 +00:00
data = json_check_results ( self . r )
2008-09-15 20:43:14 +00:00
return simplejson . dumps ( data , indent = 1 ) + " \n "
2008-09-10 02:45:17 +00:00
2008-11-07 05:35:47 +00:00
def render_summary ( self , ctx , data ) :
results = [ ]
2009-02-20 19:15:54 +00:00
if data . is_healthy ( ) :
2008-11-07 05:35:47 +00:00
results . append ( " Healthy " )
2009-02-20 19:15:54 +00:00
elif data . is_recoverable ( ) :
2008-11-07 05:35:47 +00:00
results . append ( " Not Healthy! " )
else :
results . append ( " Not Recoverable! " )
results . append ( " : " )
2009-02-20 19:15:54 +00:00
results . append ( self . _html ( data . get_summary ( ) ) )
2008-11-07 05:35:47 +00:00
return ctx . tag [ results ]
2008-10-23 23:00:31 +00:00
2008-10-30 01:09:17 +00:00
def render_repair ( self , ctx , data ) :
2009-02-20 19:15:54 +00:00
if data . is_healthy ( ) :
2008-10-30 01:09:17 +00:00
return " "
repair = T . form ( action = " . " , method = " post " ,
enctype = " multipart/form-data " ) [
T . fieldset [
T . input ( type = " hidden " , name = " t " , value = " check " ) ,
T . input ( type = " hidden " , name = " repair " , value = " true " ) ,
T . input ( type = " submit " , value = " Repair " ) ,
] ]
2009-04-09 23:59:59 +00:00
return " " # repair button disabled until we make it work correctly,
# see #622 for details
2008-10-30 01:09:17 +00:00
return ctx . tag [ repair ]
2008-07-16 22:42:56 +00:00
def render_results ( self , ctx , data ) :
2009-02-20 19:15:54 +00:00
cr = self . _render_results ( ctx , data )
2008-09-07 19:44:56 +00:00
return ctx . tag [ cr ]
2008-07-16 22:42:56 +00:00
2008-10-23 23:00:31 +00:00
class CheckAndRepairResults ( CheckerBase , rend . Page , ResultsBase ) :
2008-09-07 19:44:56 +00:00
docFactory = getxmlfile ( " check-and-repair-results.xhtml " )
2009-02-20 19:15:54 +00:00
def __init__ ( self , client , results ) :
self . client = client
2009-02-23 21:19:43 +00:00
self . r = None
if results :
self . r = ICheckAndRepairResults ( results )
2009-02-20 19:15:54 +00:00
rend . Page . __init__ ( self , results )
2008-09-07 19:44:56 +00:00
2008-09-10 02:45:17 +00:00
def json ( self , ctx ) :
inevow . IRequest ( ctx ) . setHeader ( " content-type " , " text/plain " )
2009-02-17 06:35:53 +00:00
data = json_check_and_repair_results ( self . r )
2008-09-15 20:43:14 +00:00
return simplejson . dumps ( data , indent = 1 ) + " \n "
2008-09-10 02:45:17 +00:00
2008-11-07 05:35:47 +00:00
def render_summary ( self , ctx , data ) :
2009-02-20 19:15:54 +00:00
cr = data . get_post_repair_results ( )
2008-11-07 05:35:47 +00:00
results = [ ]
2008-09-07 19:44:56 +00:00
if cr . is_healthy ( ) :
2008-11-07 05:35:47 +00:00
results . append ( " Healthy " )
elif cr . is_recoverable ( ) :
results . append ( " Not Healthy! " )
else :
results . append ( " Not Recoverable! " )
results . append ( " : " )
results . append ( self . _html ( cr . get_summary ( ) ) )
return ctx . tag [ results ]
2008-09-07 19:44:56 +00:00
def render_repair_results ( self , ctx , data ) :
2009-02-20 19:15:54 +00:00
if data . get_repair_attempted ( ) :
if data . get_repair_successful ( ) :
2008-09-07 19:44:56 +00:00
return ctx . tag [ " Repair successful " ]
else :
return ctx . tag [ " Repair unsuccessful " ]
return ctx . tag [ " No repair necessary " ]
def render_post_repair_results ( self , ctx , data ) :
2009-02-20 19:15:54 +00:00
cr = self . _render_results ( ctx , data . get_post_repair_results ( ) )
2009-01-13 01:56:19 +00:00
return ctx . tag [ T . div [ " Post-Repair Checker Results: " ] , cr ]
2008-09-07 19:44:56 +00:00
def render_maybe_pre_repair_results ( self , ctx , data ) :
2009-02-20 19:15:54 +00:00
if data . get_repair_attempted ( ) :
cr = self . _render_results ( ctx , data . get_pre_repair_results ( ) )
2008-09-07 19:44:56 +00:00
return ctx . tag [ T . div [ " Pre-Repair Checker Results: " ] , cr ]
return " "
2008-10-22 00:03:07 +00:00
class DeepCheckResults ( rend . Page , ResultsBase , ReloadMixin ) :
2008-07-17 23:47:09 +00:00
docFactory = getxmlfile ( " deep-check-results.xhtml " )
2009-02-20 19:15:54 +00:00
def __init__ ( self , client , monitor ) :
self . client = client
2008-10-22 00:03:07 +00:00
self . monitor = monitor
2008-07-17 23:47:09 +00:00
2008-10-23 23:00:31 +00:00
def childFactory ( self , ctx , name ) :
if not name :
return self
# /operation/$OPHANDLE/$STORAGEINDEX provides detailed information
# about a specific file or directory that was checked
si = base32 . a2b ( name )
r = self . monitor . get_status ( )
try :
2009-02-20 19:15:54 +00:00
return CheckResults ( self . client ,
r . get_results_for_storage_index ( si ) )
2008-10-23 23:00:31 +00:00
except KeyError :
raise WebError ( " No detailed results for SI %s " % html . escape ( name ) ,
http . NOT_FOUND )
2008-09-10 02:45:17 +00:00
def renderHTTP ( self , ctx ) :
2008-09-10 21:11:37 +00:00
if self . want_json ( ctx ) :
2008-09-10 02:45:17 +00:00
return self . json ( ctx )
return rend . Page . renderHTTP ( self , ctx )
def json ( self , ctx ) :
inevow . IRequest ( ctx ) . setHeader ( " content-type " , " text/plain " )
data = { }
2008-10-22 00:03:07 +00:00
data [ " finished " ] = self . monitor . is_finished ( )
res = self . monitor . get_status ( )
data [ " root-storage-index " ] = res . get_root_storage_index_string ( )
c = res . get_counters ( )
2008-09-10 02:45:17 +00:00
data [ " count-objects-checked " ] = c [ " count-objects-checked " ]
data [ " count-objects-healthy " ] = c [ " count-objects-healthy " ]
data [ " count-objects-unhealthy " ] = c [ " count-objects-unhealthy " ]
data [ " count-corrupt-shares " ] = c [ " count-corrupt-shares " ]
2008-09-10 21:11:37 +00:00
data [ " list-corrupt-shares " ] = [ ( idlib . nodeid_b2a ( serverid ) ,
base32 . b2a ( storage_index ) ,
2008-09-10 02:45:17 +00:00
shnum )
for ( serverid , storage_index , shnum )
2008-10-22 00:03:07 +00:00
in res . get_corrupt_shares ( ) ]
2009-02-17 06:35:53 +00:00
data [ " list-unhealthy-files " ] = [ ( path_t , json_check_results ( r ) )
2008-09-10 02:45:17 +00:00
for ( path_t , r )
2008-10-22 00:03:07 +00:00
in res . get_all_results ( ) . items ( )
2008-09-10 02:45:17 +00:00
if not r . is_healthy ( ) ]
2008-10-22 00:03:07 +00:00
data [ " stats " ] = res . get_stats ( )
2008-09-15 20:43:14 +00:00
return simplejson . dumps ( data , indent = 1 ) + " \n "
2008-09-10 02:45:17 +00:00
2008-07-17 23:47:09 +00:00
def render_root_storage_index ( self , ctx , data ) :
2008-10-22 00:03:07 +00:00
return self . monitor . get_status ( ) . get_root_storage_index_string ( )
2008-07-17 23:47:09 +00:00
def data_objects_checked ( self , ctx , data ) :
2008-10-22 00:03:07 +00:00
return self . monitor . get_status ( ) . get_counters ( ) [ " count-objects-checked " ]
2008-07-17 23:47:09 +00:00
def data_objects_healthy ( self , ctx , data ) :
2008-10-22 00:03:07 +00:00
return self . monitor . get_status ( ) . get_counters ( ) [ " count-objects-healthy " ]
2008-09-07 19:44:56 +00:00
def data_objects_unhealthy ( self , ctx , data ) :
2008-10-22 00:03:07 +00:00
return self . monitor . get_status ( ) . get_counters ( ) [ " count-objects-unhealthy " ]
2008-11-07 05:35:47 +00:00
def data_objects_unrecoverable ( self , ctx , data ) :
return self . monitor . get_status ( ) . get_counters ( ) [ " count-objects-unrecoverable " ]
2008-09-07 19:44:56 +00:00
def data_count_corrupt_shares ( self , ctx , data ) :
2008-10-22 00:03:07 +00:00
return self . monitor . get_status ( ) . get_counters ( ) [ " count-corrupt-shares " ]
2008-09-07 19:44:56 +00:00
def render_problems_p ( self , ctx , data ) :
2008-10-22 00:03:07 +00:00
c = self . monitor . get_status ( ) . get_counters ( )
2008-09-07 19:44:56 +00:00
if c [ " count-objects-unhealthy " ] :
return ctx . tag
return " "
2008-07-17 23:47:09 +00:00
def data_problems ( self , ctx , data ) :
2008-10-22 00:03:07 +00:00
all_objects = self . monitor . get_status ( ) . get_all_results ( )
2008-09-07 19:44:56 +00:00
for path in sorted ( all_objects . keys ( ) ) :
cr = all_objects [ path ]
2009-01-06 20:37:03 +00:00
assert ICheckResults . providedBy ( cr )
2008-09-07 19:44:56 +00:00
if not cr . is_healthy ( ) :
yield path , cr
2008-08-12 04:03:26 +00:00
def render_problem ( self , ctx , data ) :
2008-09-07 19:44:56 +00:00
path , cr = data
summary_text = " "
summary = cr . get_summary ( )
if summary :
summary_text = " : " + summary
summary_text + = " [SI: %s ] " % cr . get_storage_index_string ( )
2009-01-13 01:56:19 +00:00
return ctx . tag [ self . _join_pathstring ( path ) , self . _html ( summary_text ) ]
2008-09-07 19:44:56 +00:00
def render_servers_with_corrupt_shares_p ( self , ctx , data ) :
2008-10-22 00:03:07 +00:00
if self . monitor . get_status ( ) . get_counters ( ) [ " count-corrupt-shares " ] :
2008-09-07 19:44:56 +00:00
return ctx . tag
return " "
def data_servers_with_corrupt_shares ( self , ctx , data ) :
servers = [ serverid
for ( serverid , storage_index , sharenum )
2008-10-22 00:03:07 +00:00
in self . monitor . get_status ( ) . get_corrupt_shares ( ) ]
2008-09-07 19:44:56 +00:00
servers . sort ( )
return servers
def render_server_problem ( self , ctx , data ) :
serverid = data
data = [ idlib . shortnodeid_b2a ( serverid ) ]
2009-06-02 03:07:50 +00:00
sb = self . client . get_storage_broker ( )
nickname = sb . get_nickname_for_serverid ( serverid )
2008-09-07 19:44:56 +00:00
if nickname :
data . append ( " ( %s ) " % self . _html ( nickname ) )
return ctx . tag [ data ]
def render_corrupt_shares_p ( self , ctx , data ) :
2008-10-22 00:03:07 +00:00
if self . monitor . get_status ( ) . get_counters ( ) [ " count-corrupt-shares " ] :
2008-09-07 19:44:56 +00:00
return ctx . tag
return " "
def data_corrupt_shares ( self , ctx , data ) :
2008-10-22 00:03:07 +00:00
return self . monitor . get_status ( ) . get_corrupt_shares ( )
2008-09-07 19:44:56 +00:00
def render_share_problem ( self , ctx , data ) :
serverid , storage_index , sharenum = data
2009-06-02 03:07:50 +00:00
sb = self . client . get_storage_broker ( )
nickname = sb . get_nickname_for_serverid ( serverid )
2008-09-07 19:44:56 +00:00
ctx . fillSlots ( " serverid " , idlib . shortnodeid_b2a ( serverid ) )
if nickname :
ctx . fillSlots ( " nickname " , self . _html ( nickname ) )
2008-10-23 23:00:31 +00:00
ctx . fillSlots ( " si " , self . _render_si_link ( ctx , storage_index ) )
2008-09-07 19:44:56 +00:00
ctx . fillSlots ( " shnum " , str ( sharenum ) )
return ctx . tag
def render_return ( self , ctx , data ) :
req = inevow . IRequest ( ctx )
return_to = get_arg ( req , " return_to " , None )
if return_to :
2009-04-07 18:54:59 +00:00
return T . div [ T . a ( href = return_to ) [ " Return to file/directory. " ] ]
2008-09-07 19:44:56 +00:00
return " "
2008-08-12 04:03:26 +00:00
def data_all_objects ( self , ctx , data ) :
2008-10-22 00:03:07 +00:00
r = self . monitor . get_status ( ) . get_all_results ( )
2008-09-07 19:44:56 +00:00
for path in sorted ( r . keys ( ) ) :
yield ( path , r [ path ] )
2008-08-12 04:03:26 +00:00
def render_object ( self , ctx , data ) :
2008-09-07 19:44:56 +00:00
path , r = data
2009-01-13 01:56:19 +00:00
ctx . fillSlots ( " path " , self . _join_pathstring ( path ) )
2008-08-12 04:03:26 +00:00
ctx . fillSlots ( " healthy " , str ( r . is_healthy ( ) ) )
2008-11-07 05:35:47 +00:00
ctx . fillSlots ( " recoverable " , str ( r . is_recoverable ( ) ) )
2008-10-23 23:00:31 +00:00
storage_index = r . get_storage_index ( )
ctx . fillSlots ( " storage_index " , self . _render_si_link ( ctx , storage_index ) )
2008-09-07 19:44:56 +00:00
ctx . fillSlots ( " summary " , self . _html ( r . get_summary ( ) ) )
2008-08-12 04:03:26 +00:00
return ctx . tag
2008-07-17 23:47:09 +00:00
2008-09-07 19:44:56 +00:00
def render_runtime ( self , ctx , data ) :
req = inevow . IRequest ( ctx )
runtime = time . time ( ) - req . processing_started_timestamp
return ctx . tag [ " runtime: %s seconds " % runtime ]
2008-10-22 00:03:07 +00:00
class DeepCheckAndRepairResults ( rend . Page , ResultsBase , ReloadMixin ) :
2008-09-07 19:44:56 +00:00
docFactory = getxmlfile ( " deep-check-and-repair-results.xhtml " )
2009-02-20 19:15:54 +00:00
def __init__ ( self , client , monitor ) :
self . client = client
2008-10-22 00:03:07 +00:00
self . monitor = monitor
2008-09-07 19:44:56 +00:00
2009-01-13 01:56:19 +00:00
def childFactory ( self , ctx , name ) :
if not name :
return self
# /operation/$OPHANDLE/$STORAGEINDEX provides detailed information
# about a specific file or directory that was checked
si = base32 . a2b ( name )
r = self . monitor . get_status ( )
try :
2009-02-20 19:15:54 +00:00
return CheckAndRepairResults ( self . client ,
r . get_results_for_storage_index ( si ) )
2009-01-13 01:56:19 +00:00
except KeyError :
raise WebError ( " No detailed results for SI %s " % html . escape ( name ) ,
http . NOT_FOUND )
2008-09-10 02:45:17 +00:00
def renderHTTP ( self , ctx ) :
2008-09-10 21:11:37 +00:00
if self . want_json ( ctx ) :
2008-09-10 02:45:17 +00:00
return self . json ( ctx )
return rend . Page . renderHTTP ( self , ctx )
def json ( self , ctx ) :
inevow . IRequest ( ctx ) . setHeader ( " content-type " , " text/plain " )
2008-10-22 00:03:07 +00:00
res = self . monitor . get_status ( )
2008-09-10 02:45:17 +00:00
data = { }
2008-10-22 00:03:07 +00:00
data [ " finished " ] = self . monitor . is_finished ( )
data [ " root-storage-index " ] = res . get_root_storage_index_string ( )
c = res . get_counters ( )
2008-09-10 02:45:17 +00:00
data [ " count-objects-checked " ] = c [ " count-objects-checked " ]
data [ " count-objects-healthy-pre-repair " ] = c [ " count-objects-healthy-pre-repair " ]
data [ " count-objects-unhealthy-pre-repair " ] = c [ " count-objects-unhealthy-pre-repair " ]
data [ " count-objects-healthy-post-repair " ] = c [ " count-objects-healthy-post-repair " ]
data [ " count-objects-unhealthy-post-repair " ] = c [ " count-objects-unhealthy-post-repair " ]
data [ " count-repairs-attempted " ] = c [ " count-repairs-attempted " ]
data [ " count-repairs-successful " ] = c [ " count-repairs-successful " ]
data [ " count-repairs-unsuccessful " ] = c [ " count-repairs-unsuccessful " ]
data [ " count-corrupt-shares-pre-repair " ] = c [ " count-corrupt-shares-pre-repair " ]
data [ " count-corrupt-shares-post-repair " ] = c [ " count-corrupt-shares-pre-repair " ]
2008-09-10 21:11:37 +00:00
data [ " list-corrupt-shares " ] = [ ( idlib . nodeid_b2a ( serverid ) ,
base32 . b2a ( storage_index ) ,
2008-09-10 02:45:17 +00:00
shnum )
for ( serverid , storage_index , shnum )
2008-10-22 00:03:07 +00:00
in res . get_corrupt_shares ( ) ]
2008-09-10 02:45:17 +00:00
2008-11-19 02:12:10 +00:00
remaining_corrupt = [ ( idlib . nodeid_b2a ( serverid ) ,
base32 . b2a ( storage_index ) ,
shnum )
for ( serverid , storage_index , shnum )
in res . get_remaining_corrupt_shares ( ) ]
data [ " list-remaining-corrupt-shares " ] = remaining_corrupt
unhealthy = [ ( path_t ,
2009-02-17 06:35:53 +00:00
json_check_results ( crr . get_pre_repair_results ( ) ) )
2008-11-19 02:12:10 +00:00
for ( path_t , crr )
in res . get_all_results ( ) . items ( )
if not crr . get_pre_repair_results ( ) . is_healthy ( ) ]
data [ " list-unhealthy-files " ] = unhealthy
2008-10-22 00:03:07 +00:00
data [ " stats " ] = res . get_stats ( )
2008-09-15 20:43:14 +00:00
return simplejson . dumps ( data , indent = 1 ) + " \n "
2008-09-10 02:45:17 +00:00
2008-09-07 19:44:56 +00:00
def render_root_storage_index ( self , ctx , data ) :
2008-10-22 00:03:07 +00:00
return self . monitor . get_status ( ) . get_root_storage_index_string ( )
2008-09-07 19:44:56 +00:00
def data_objects_checked ( self , ctx , data ) :
2008-10-22 00:03:07 +00:00
return self . monitor . get_status ( ) . get_counters ( ) [ " count-objects-checked " ]
2008-09-07 19:44:56 +00:00
def data_objects_healthy ( self , ctx , data ) :
2008-10-22 00:03:07 +00:00
return self . monitor . get_status ( ) . get_counters ( ) [ " count-objects-healthy-pre-repair " ]
2008-09-07 19:44:56 +00:00
def data_objects_unhealthy ( self , ctx , data ) :
2008-10-22 00:03:07 +00:00
return self . monitor . get_status ( ) . get_counters ( ) [ " count-objects-unhealthy-pre-repair " ]
2008-09-07 19:44:56 +00:00
def data_corrupt_shares ( self , ctx , data ) :
2008-10-22 00:03:07 +00:00
return self . monitor . get_status ( ) . get_counters ( ) [ " count-corrupt-shares-pre-repair " ]
2008-09-07 19:44:56 +00:00
def data_repairs_attempted ( self , ctx , data ) :
2008-10-22 00:03:07 +00:00
return self . monitor . get_status ( ) . get_counters ( ) [ " count-repairs-attempted " ]
2008-09-07 19:44:56 +00:00
def data_repairs_successful ( self , ctx , data ) :
2008-10-22 00:03:07 +00:00
return self . monitor . get_status ( ) . get_counters ( ) [ " count-repairs-successful " ]
2008-09-07 19:44:56 +00:00
def data_repairs_unsuccessful ( self , ctx , data ) :
2008-10-22 00:03:07 +00:00
return self . monitor . get_status ( ) . get_counters ( ) [ " count-repairs-unsuccessful " ]
2008-09-07 19:44:56 +00:00
def data_objects_healthy_post ( self , ctx , data ) :
2008-10-22 00:03:07 +00:00
return self . monitor . get_status ( ) . get_counters ( ) [ " count-objects-healthy-post-repair " ]
2008-09-07 19:44:56 +00:00
def data_objects_unhealthy_post ( self , ctx , data ) :
2008-10-22 00:03:07 +00:00
return self . monitor . get_status ( ) . get_counters ( ) [ " count-objects-unhealthy-post-repair " ]
2008-09-07 19:44:56 +00:00
def data_corrupt_shares_post ( self , ctx , data ) :
2008-10-22 00:03:07 +00:00
return self . monitor . get_status ( ) . get_counters ( ) [ " count-corrupt-shares-post-repair " ]
2008-09-07 19:44:56 +00:00
def render_pre_repair_problems_p ( self , ctx , data ) :
2008-10-22 00:03:07 +00:00
c = self . monitor . get_status ( ) . get_counters ( )
2008-09-07 19:44:56 +00:00
if c [ " count-objects-unhealthy-pre-repair " ] :
return ctx . tag
return " "
def data_pre_repair_problems ( self , ctx , data ) :
2008-10-22 00:03:07 +00:00
all_objects = self . monitor . get_status ( ) . get_all_results ( )
2008-09-07 19:44:56 +00:00
for path in sorted ( all_objects . keys ( ) ) :
r = all_objects [ path ]
assert ICheckAndRepairResults . providedBy ( r )
cr = r . get_pre_repair_results ( )
if not cr . is_healthy ( ) :
yield path , cr
def render_problem ( self , ctx , data ) :
path , cr = data
2009-01-13 01:56:19 +00:00
return ctx . tag [ self . _join_pathstring ( path ) , " : " ,
self . _html ( cr . get_summary ( ) ) ]
2008-09-07 19:44:56 +00:00
def render_post_repair_problems_p ( self , ctx , data ) :
2008-10-22 00:03:07 +00:00
c = self . monitor . get_status ( ) . get_counters ( )
2008-09-07 19:44:56 +00:00
if ( c [ " count-objects-unhealthy-post-repair " ]
or c [ " count-corrupt-shares-post-repair " ] ) :
return ctx . tag
return " "
def data_post_repair_problems ( self , ctx , data ) :
2008-10-22 00:03:07 +00:00
all_objects = self . monitor . get_status ( ) . get_all_results ( )
2008-09-07 19:44:56 +00:00
for path in sorted ( all_objects . keys ( ) ) :
r = all_objects [ path ]
assert ICheckAndRepairResults . providedBy ( r )
cr = r . get_post_repair_results ( )
if not cr . is_healthy ( ) :
yield path , cr
def render_servers_with_corrupt_shares_p ( self , ctx , data ) :
2008-10-22 00:03:07 +00:00
if self . monitor . get_status ( ) . get_counters ( ) [ " count-corrupt-shares-pre-repair " ] :
2008-09-07 19:44:56 +00:00
return ctx . tag
return " "
def data_servers_with_corrupt_shares ( self , ctx , data ) :
return [ ] # TODO
def render_server_problem ( self , ctx , data ) :
pass
def render_remaining_corrupt_shares_p ( self , ctx , data ) :
2008-10-22 00:03:07 +00:00
if self . monitor . get_status ( ) . get_counters ( ) [ " count-corrupt-shares-post-repair " ] :
2008-09-07 19:44:56 +00:00
return ctx . tag
return " "
def data_post_repair_corrupt_shares ( self , ctx , data ) :
return [ ] # TODO
def render_share_problem ( self , ctx , data ) :
pass
2008-07-17 23:47:09 +00:00
def render_return ( self , ctx , data ) :
req = inevow . IRequest ( ctx )
return_to = get_arg ( req , " return_to " , None )
if return_to :
2009-04-07 18:54:59 +00:00
return T . div [ T . a ( href = return_to ) [ " Return to file/directory. " ] ]
2008-07-17 23:47:09 +00:00
return " "
2008-08-13 03:34:26 +00:00
2008-09-07 19:44:56 +00:00
def data_all_objects ( self , ctx , data ) :
2008-10-22 00:03:07 +00:00
r = self . monitor . get_status ( ) . get_all_results ( )
2008-09-07 19:44:56 +00:00
for path in sorted ( r . keys ( ) ) :
yield ( path , r [ path ] )
def render_object ( self , ctx , data ) :
path , r = data
2009-01-13 01:56:19 +00:00
ctx . fillSlots ( " path " , self . _join_pathstring ( path ) )
2008-09-07 19:44:56 +00:00
ctx . fillSlots ( " healthy_pre_repair " ,
str ( r . get_pre_repair_results ( ) . is_healthy ( ) ) )
2009-01-13 01:56:19 +00:00
ctx . fillSlots ( " recoverable_pre_repair " ,
str ( r . get_pre_repair_results ( ) . is_recoverable ( ) ) )
2008-09-07 19:44:56 +00:00
ctx . fillSlots ( " healthy_post_repair " ,
str ( r . get_post_repair_results ( ) . is_healthy ( ) ) )
2009-01-13 01:56:19 +00:00
storage_index = r . get_storage_index ( )
ctx . fillSlots ( " storage_index " ,
self . _render_si_link ( ctx , storage_index ) )
2008-09-07 19:44:56 +00:00
ctx . fillSlots ( " summary " ,
self . _html ( r . get_pre_repair_results ( ) . get_summary ( ) ) )
return ctx . tag
2008-08-13 03:34:26 +00:00
def render_runtime ( self , ctx , data ) :
req = inevow . IRequest ( ctx )
runtime = time . time ( ) - req . processing_started_timestamp
return ctx . tag [ " runtime: %s seconds " % runtime ]