2008-10-29 22:34:31 +00:00
import os . path , re , urllib
2007-12-05 06:42:54 +00:00
import simplejson
2009-02-17 05:12:42 +00:00
from StringIO import StringIO
2007-07-07 02:43:55 +00:00
from twisted . application import service
from twisted . trial import unittest
2008-05-20 18:47:43 +00:00
from twisted . internet import defer , reactor
2007-07-17 19:16:45 +00:00
from twisted . web import client , error , http
2007-07-08 03:06:58 +00:00
from twisted . python import failure , log
2009-03-04 04:56:30 +00:00
from nevow import rend
2009-02-18 21:46:55 +00:00
from allmydata import interfaces , uri , webish
2009-03-07 05:45:17 +00:00
from allmydata . storage . shares import get_share_file
2009-06-01 21:06:04 +00:00
from allmydata . storage_client import StorageFarmBroker
2008-07-16 20:14:39 +00:00
from allmydata . immutable import upload , download
2008-05-20 22:21:46 +00:00
from allmydata . web import status , common
2009-02-17 05:12:42 +00:00
from allmydata . scripts . debug import CorruptShareOptions , corrupt_share
2008-10-29 04:28:31 +00:00
from allmydata . util import fileutil , base32
2008-12-19 15:39:24 +00:00
from allmydata . util . assertutil import precondition
2008-04-17 20:02:22 +00:00
from allmydata . test . common import FakeDirectoryNode , FakeCHKFileNode , \
2009-02-24 22:40:17 +00:00
FakeMutableFileNode , create_chk_filenode , WebErrorMixin , ShouldFailMixin
2008-04-17 20:02:22 +00:00
from allmydata . interfaces import IURI , INewDirectoryURI , \
IReadonlyNewDirectoryURI , IFileURI , IMutableFileURI , IMutableFileNode
from allmydata . mutable import servermap , publish , retrieve
2008-10-29 04:28:31 +00:00
import common_util as testutil
2009-02-17 05:12:42 +00:00
from allmydata . test . no_network import GridTestMixin
2007-12-03 21:52:42 +00:00
2009-02-25 01:33:00 +00:00
from allmydata . test . common_web import HTTPClientGETFactory , \
HTTPClientHEADFactory
2007-07-07 02:43:55 +00:00
# create a fake uploader/downloader, and a couple of fake dirnodes, then
# create a webserver that works against them
2009-06-09 02:17:53 +00:00
timeout = 480 # Most of these take longer than 240 seconds on Francois's arm box.
2009-06-08 22:55:57 +00:00
2009-02-20 19:15:54 +00:00
class FakeStatsProvider :
def get_stats ( self ) :
stats = { ' stats ' : { } , ' counters ' : { } }
return stats
2007-12-07 00:17:02 +00:00
class FakeClient ( service . MultiService ) :
2007-07-07 02:43:55 +00:00
nodeid = " fake_nodeid "
2008-06-03 22:02:10 +00:00
nickname = " fake_nickname "
2007-08-22 21:54:34 +00:00
basedir = " fake_basedir "
2007-07-07 02:43:55 +00:00
def get_versions ( self ) :
return { ' allmydata ' : " fake " ,
' foolscap ' : " fake " ,
' twisted ' : " fake " ,
' zfec ' : " fake " ,
}
introducer_furl = " None "
2009-06-23 02:10:47 +00:00
2008-03-03 21:48:52 +00:00
_all_upload_status = [ upload . UploadStatus ( ) ]
_all_download_status = [ download . DownloadStatus ( ) ]
2008-04-17 20:02:22 +00:00
_all_mapupdate_statuses = [ servermap . UpdateStatus ( ) ]
_all_publish_statuses = [ publish . PublishStatus ( ) ]
_all_retrieve_statuses = [ retrieve . RetrieveStatus ( ) ]
2008-03-24 16:46:06 +00:00
convergence = " some random string "
2009-02-20 19:15:54 +00:00
stats_provider = FakeStatsProvider ( )
2008-03-01 06:03:00 +00:00
2007-07-07 02:43:55 +00:00
def connected_to_introducer ( self ) :
return False
2009-06-23 02:10:47 +00:00
storage_broker = StorageFarmBroker ( None , permute_peers = True )
2009-06-02 02:25:11 +00:00
def get_storage_broker ( self ) :
return self . storage_broker
2008-12-05 09:12:10 +00:00
2009-07-02 22:25:37 +00:00
def create_node_from_uri ( self , auri , readcap = None ) :
if not auri :
auri = readcap
2008-12-19 15:39:24 +00:00
precondition ( isinstance ( auri , str ) , auri )
2008-03-01 01:40:27 +00:00
u = uri . from_string ( auri )
2007-12-05 06:01:37 +00:00
if ( INewDirectoryURI . providedBy ( u )
or IReadonlyNewDirectoryURI . providedBy ( u ) ) :
2008-03-01 01:40:27 +00:00
return FakeDirectoryNode ( self ) . init_from_uri ( u )
2007-12-05 06:01:37 +00:00
if IFileURI . providedBy ( u ) :
return FakeCHKFileNode ( u , self )
assert IMutableFileURI . providedBy ( u ) , u
return FakeMutableFileNode ( self ) . init_from_uri ( u )
2007-12-04 04:37:54 +00:00
2008-01-14 21:55:59 +00:00
def create_empty_dirnode ( self ) :
2008-03-01 01:40:27 +00:00
n = FakeDirectoryNode ( self )
2008-01-14 21:55:59 +00:00
d = n . create ( )
2007-12-05 06:01:37 +00:00
d . addCallback ( lambda res : n )
return d
2007-12-03 21:52:42 +00:00
2008-06-03 07:03:16 +00:00
MUTABLE_SIZELIMIT = FakeMutableFileNode . MUTABLE_SIZELIMIT
2008-01-14 21:55:59 +00:00
def create_mutable_file ( self , contents = " " ) :
2007-12-05 06:01:37 +00:00
n = FakeMutableFileNode ( self )
return n . create ( contents )
2007-07-07 02:43:55 +00:00
2008-01-14 21:55:59 +00:00
def upload ( self , uploadable ) :
2007-07-20 01:21:44 +00:00
d = uploadable . get_size ( )
d . addCallback ( lambda size : uploadable . read ( size ) )
2007-12-05 06:01:37 +00:00
def _got_data ( datav ) :
data = " " . join ( datav )
n = create_chk_filenode ( self , data )
2008-02-06 04:01:38 +00:00
results = upload . UploadResults ( )
results . uri = n . get_uri ( )
return results
2007-07-20 01:21:44 +00:00
d . addCallback ( _got_data )
return d
2007-07-07 02:43:55 +00:00
2008-04-17 20:02:22 +00:00
def list_all_upload_statuses ( self ) :
2008-03-03 21:48:52 +00:00
return self . _all_upload_status
2008-04-17 20:02:22 +00:00
def list_all_download_statuses ( self ) :
2008-03-03 21:48:52 +00:00
return self . _all_download_status
2008-04-17 20:02:22 +00:00
def list_all_mapupdate_statuses ( self ) :
return self . _all_mapupdate_statuses
def list_all_publish_statuses ( self ) :
return self . _all_publish_statuses
def list_all_retrieve_statuses ( self ) :
return self . _all_retrieve_statuses
def list_all_helper_statuses ( self ) :
2008-04-15 01:36:27 +00:00
return [ ]
2007-12-05 06:01:37 +00:00
2007-08-10 15:40:02 +00:00
class WebMixin ( object ) :
2007-07-07 02:43:55 +00:00
def setUp ( self ) :
2007-12-07 00:17:02 +00:00
self . s = FakeClient ( )
2007-07-07 02:43:55 +00:00
self . s . startService ( )
2008-10-29 22:34:31 +00:00
self . staticdir = self . mktemp ( )
2009-02-20 19:15:54 +00:00
self . ws = s = webish . WebishServer ( self . s , " 0 " , staticdir = self . staticdir )
2007-07-07 02:43:55 +00:00
s . setServiceParent ( self . s )
2008-05-20 18:47:43 +00:00
self . webish_port = port = s . listener . _port . getHost ( ) . port
2007-07-07 02:43:55 +00:00
self . webish_url = " http://localhost: %d " % port
2007-12-03 21:52:42 +00:00
l = [ self . s . create_empty_dirnode ( ) for x in range ( 6 ) ]
d = defer . DeferredList ( l )
def _then ( res ) :
self . public_root = res [ 0 ] [ 1 ]
assert interfaces . IDirectoryNode . providedBy ( self . public_root ) , res
self . public_url = " /uri/ " + self . public_root . get_uri ( )
self . private_root = res [ 1 ] [ 1 ]
foo = res [ 2 ] [ 1 ]
self . _foo_node = foo
self . _foo_uri = foo . get_uri ( )
self . _foo_readonly_uri = foo . get_readonly_uri ( )
2009-02-04 02:22:48 +00:00
self . _foo_verifycap = foo . get_verify_cap ( ) . to_string ( )
2007-12-05 06:01:37 +00:00
# NOTE: we ignore the deferred on all set_uri() calls, because we
# know the fake nodes do these synchronously
2008-02-14 22:45:56 +00:00
self . public_root . set_uri ( u " foo " , foo . get_uri ( ) )
2007-12-03 21:52:42 +00:00
2007-12-05 06:01:37 +00:00
self . BAR_CONTENTS , n , self . _bar_txt_uri = self . makefile ( 0 )
2008-02-14 22:45:56 +00:00
foo . set_uri ( u " bar.txt " , self . _bar_txt_uri )
2009-02-04 02:22:48 +00:00
self . _bar_txt_verifycap = n . get_verify_cap ( ) . to_string ( )
2008-02-12 02:14:10 +00:00
2008-02-14 22:45:56 +00:00
foo . set_uri ( u " empty " , res [ 3 ] [ 1 ] . get_uri ( ) )
2007-12-03 21:52:42 +00:00
sub_uri = res [ 4 ] [ 1 ] . get_uri ( )
2008-06-18 02:49:40 +00:00
self . _sub_uri = sub_uri
2008-02-14 22:45:56 +00:00
foo . set_uri ( u " sub " , sub_uri )
2007-12-05 06:01:37 +00:00
sub = self . s . create_node_from_uri ( sub_uri )
2007-12-03 21:52:42 +00:00
2007-12-05 06:01:37 +00:00
_ign , n , blocking_uri = self . makefile ( 1 )
2008-02-14 22:45:56 +00:00
foo . set_uri ( u " blockingfile " , blocking_uri )
unicode_filename = u " n \u00fc .txt " # n u-umlaut . t x t
# ok, unicode calls it LATIN SMALL LETTER U WITH DIAERESIS but I
# still think of it as an umlaut
foo . set_uri ( unicode_filename , self . _bar_txt_uri )
2007-12-03 21:52:42 +00:00
2007-12-05 06:01:37 +00:00
_ign , n , baz_file = self . makefile ( 2 )
2009-01-23 05:01:36 +00:00
self . _baz_file_uri = baz_file
2008-02-14 22:45:56 +00:00
sub . set_uri ( u " baz.txt " , baz_file )
2007-12-03 21:52:42 +00:00
2007-12-05 06:01:37 +00:00
_ign , n , self . _bad_file_uri = self . makefile ( 3 )
# this uri should not be downloadable
del FakeCHKFileNode . all_contents [ self . _bad_file_uri ]
2007-12-03 21:52:42 +00:00
rodir = res [ 5 ] [ 1 ]
2008-02-14 22:45:56 +00:00
self . public_root . set_uri ( u " reedownlee " , rodir . get_readonly_uri ( ) )
rodir . set_uri ( u " nor " , baz_file )
2007-12-03 21:52:42 +00:00
# public/
# public/foo/
# public/foo/bar.txt
# public/foo/blockingfile
# public/foo/empty/
# public/foo/sub/
# public/foo/sub/baz.txt
# public/reedownlee/
# public/reedownlee/nor
self . NEWFILE_CONTENTS = " newfile contents \n "
2008-02-12 02:14:10 +00:00
2008-02-14 22:45:56 +00:00
return foo . get_metadata_for ( u " bar.txt " )
2007-12-03 21:52:42 +00:00
d . addCallback ( _then )
2008-02-12 02:14:10 +00:00
def _got_metadata ( metadata ) :
self . _bar_txt_metadata = metadata
d . addCallback ( _got_metadata )
2007-12-03 21:52:42 +00:00
return d
2007-07-07 02:43:55 +00:00
2007-07-07 07:16:36 +00:00
def makefile ( self , number ) :
2007-12-05 06:01:37 +00:00
contents = " contents of file %s \n " % number
n = create_chk_filenode ( self . s , contents )
return contents , n , n . get_uri ( )
2007-07-12 23:17:49 +00:00
2007-07-07 02:43:55 +00:00
def tearDown ( self ) :
return self . s . stopService ( )
def failUnlessIsBarDotTxt ( self , res ) :
2008-02-14 22:45:56 +00:00
self . failUnlessEqual ( res , self . BAR_CONTENTS , res )
2007-07-07 02:43:55 +00:00
2007-07-08 07:17:11 +00:00
def failUnlessIsBarJSON ( self , res ) :
2007-12-05 06:42:54 +00:00
data = simplejson . loads ( res )
2007-07-08 07:17:11 +00:00
self . failUnless ( isinstance ( data , list ) )
2008-09-30 22:21:06 +00:00
self . failUnlessEqual ( data [ 0 ] , u " filenode " )
2007-07-08 07:17:11 +00:00
self . failUnless ( isinstance ( data [ 1 ] , dict ) )
2008-05-20 22:40:49 +00:00
self . failIf ( data [ 1 ] [ " mutable " ] )
2007-08-23 20:00:39 +00:00
self . failIf ( " rw_uri " in data [ 1 ] ) # immutable
self . failUnlessEqual ( data [ 1 ] [ " ro_uri " ] , self . _bar_txt_uri )
2009-02-04 02:22:48 +00:00
self . failUnlessEqual ( data [ 1 ] [ " verify_uri " ] , self . _bar_txt_verifycap )
2007-12-05 06:01:37 +00:00
self . failUnlessEqual ( data [ 1 ] [ " size " ] , len ( self . BAR_CONTENTS ) )
2007-07-08 07:17:11 +00:00
2007-07-08 03:06:58 +00:00
def failUnlessIsFooJSON ( self , res ) :
2007-12-05 06:42:54 +00:00
data = simplejson . loads ( res )
2007-07-08 07:17:11 +00:00
self . failUnless ( isinstance ( data , list ) )
2007-12-03 21:52:42 +00:00
self . failUnlessEqual ( data [ 0 ] , " dirnode " , res )
2007-07-08 07:17:11 +00:00
self . failUnless ( isinstance ( data [ 1 ] , dict ) )
2008-05-20 22:40:49 +00:00
self . failUnless ( data [ 1 ] [ " mutable " ] )
2007-08-23 20:00:39 +00:00
self . failUnless ( " rw_uri " in data [ 1 ] ) # mutable
self . failUnlessEqual ( data [ 1 ] [ " rw_uri " ] , self . _foo_uri )
self . failUnlessEqual ( data [ 1 ] [ " ro_uri " ] , self . _foo_readonly_uri )
2009-02-04 02:22:48 +00:00
self . failUnlessEqual ( data [ 1 ] [ " verify_uri " ] , self . _foo_verifycap )
2007-12-04 04:37:54 +00:00
2008-09-30 22:21:06 +00:00
kidnames = sorted ( [ unicode ( n ) for n in data [ 1 ] [ " children " ] ] )
2007-07-08 07:17:11 +00:00
self . failUnlessEqual ( kidnames ,
2008-02-14 22:45:56 +00:00
[ u " bar.txt " , u " blockingfile " , u " empty " ,
u " n \u00fc .txt " , u " sub " ] )
2008-09-30 22:21:06 +00:00
kids = dict ( [ ( unicode ( name ) , value )
for ( name , value )
in data [ 1 ] [ " children " ] . iteritems ( ) ] )
2008-02-14 22:45:56 +00:00
self . failUnlessEqual ( kids [ u " sub " ] [ 0 ] , " dirnode " )
self . failUnless ( " metadata " in kids [ u " sub " ] [ 1 ] )
self . failUnless ( " ctime " in kids [ u " sub " ] [ 1 ] [ " metadata " ] )
self . failUnless ( " mtime " in kids [ u " sub " ] [ 1 ] [ " metadata " ] )
self . failUnlessEqual ( kids [ u " bar.txt " ] [ 0 ] , " filenode " )
self . failUnlessEqual ( kids [ u " bar.txt " ] [ 1 ] [ " size " ] , len ( self . BAR_CONTENTS ) )
self . failUnlessEqual ( kids [ u " bar.txt " ] [ 1 ] [ " ro_uri " ] , self . _bar_txt_uri )
2009-02-04 02:22:48 +00:00
self . failUnlessEqual ( kids [ u " bar.txt " ] [ 1 ] [ " verify_uri " ] ,
self . _bar_txt_verifycap )
2008-02-14 22:45:56 +00:00
self . failUnlessEqual ( kids [ u " bar.txt " ] [ 1 ] [ " metadata " ] [ " ctime " ] ,
2008-02-12 02:14:10 +00:00
self . _bar_txt_metadata [ " ctime " ] )
2008-02-14 22:45:56 +00:00
self . failUnlessEqual ( kids [ u " n \u00fc .txt " ] [ 1 ] [ " ro_uri " ] ,
self . _bar_txt_uri )
2007-07-08 03:06:58 +00:00
2008-10-28 20:41:04 +00:00
def GET ( self , urlpath , followRedirect = False , return_response = False ,
* * kwargs ) :
# if return_response=True, this fires with (data, statuscode,
# respheaders) instead of just data.
2008-07-19 01:58:57 +00:00
assert not isinstance ( urlpath , unicode )
2007-07-07 02:43:55 +00:00
url = self . webish_url + urlpath
2008-10-28 20:41:04 +00:00
factory = HTTPClientGETFactory ( url , method = " GET " ,
followRedirect = followRedirect , * * kwargs )
reactor . connectTCP ( " localhost " , self . webish_port , factory )
d = factory . deferred
def _got_data ( data ) :
return ( data , factory . status , factory . response_headers )
if return_response :
d . addCallback ( _got_data )
return factory . deferred
2007-07-07 02:43:55 +00:00
2008-10-28 20:41:04 +00:00
def HEAD ( self , urlpath , return_response = False , * * kwargs ) :
2008-05-20 18:47:43 +00:00
# this requires some surgery, because twisted.web.client doesn't want
# to give us back the response headers.
2008-10-28 20:41:04 +00:00
factory = HTTPClientHEADFactory ( urlpath , method = " HEAD " , * * kwargs )
2008-05-20 18:47:43 +00:00
reactor . connectTCP ( " localhost " , self . webish_port , factory )
2008-10-28 20:41:04 +00:00
d = factory . deferred
def _got_data ( data ) :
return ( data , factory . status , factory . response_headers )
if return_response :
d . addCallback ( _got_data )
2008-05-20 18:47:43 +00:00
return factory . deferred
2008-10-28 20:41:04 +00:00
def PUT ( self , urlpath , data , * * kwargs ) :
2007-07-07 02:43:55 +00:00
url = self . webish_url + urlpath
2008-10-28 20:41:04 +00:00
return client . getPage ( url , method = " PUT " , postdata = data , * * kwargs )
2007-07-07 02:43:55 +00:00
def DELETE ( self , urlpath ) :
url = self . webish_url + urlpath
return client . getPage ( url , method = " DELETE " )
2007-12-15 00:52:05 +00:00
def POST ( self , urlpath , followRedirect = False , * * fields ) :
2007-07-07 02:43:55 +00:00
url = self . webish_url + urlpath
2007-07-08 03:06:58 +00:00
sepbase = " boogabooga "
sep = " -- " + sepbase
form = [ ]
form . append ( sep )
form . append ( ' Content-Disposition: form-data; name= " _charset " ' )
form . append ( ' ' )
form . append ( ' UTF-8 ' )
form . append ( sep )
for name , value in fields . iteritems ( ) :
if isinstance ( value , tuple ) :
filename , value = value
form . append ( ' Content-Disposition: form-data; name= " %s " ; '
2008-02-14 22:45:56 +00:00
' filename= " %s " ' % ( name , filename . encode ( " utf-8 " ) ) )
2007-07-08 03:06:58 +00:00
else :
form . append ( ' Content-Disposition: form-data; name= " %s " ' % name )
form . append ( ' ' )
2008-06-04 00:09:39 +00:00
if isinstance ( value , unicode ) :
value = value . encode ( " utf-8 " )
else :
value = str ( value )
assert isinstance ( value , str )
form . append ( value )
2007-07-08 03:06:58 +00:00
form . append ( sep )
form [ - 1 ] + = " -- "
body = " \r \n " . join ( form ) + " \r \n "
headers = { " content-type " : " multipart/form-data; boundary= %s " % sepbase ,
}
return client . getPage ( url , method = " POST " , postdata = body ,
2007-12-15 00:52:05 +00:00
headers = headers , followRedirect = followRedirect )
2007-07-07 02:43:55 +00:00
2007-07-16 18:53:12 +00:00
def shouldFail ( self , res , expected_failure , which ,
substring = None , response_substring = None ) :
2007-07-07 02:43:55 +00:00
if isinstance ( res , failure . Failure ) :
res . trap ( expected_failure )
if substring :
self . failUnless ( substring in str ( res ) ,
" substring ' %s ' not in ' %s ' "
% ( substring , str ( res ) ) )
2007-07-16 18:53:12 +00:00
if response_substring :
self . failUnless ( response_substring in res . value . response ,
2008-05-19 19:33:39 +00:00
" response substring ' %s ' not in ' %s ' "
2007-07-16 18:53:12 +00:00
% ( response_substring , res . value . response ) )
2007-07-07 02:43:55 +00:00
else :
self . fail ( " %s was supposed to raise %s , not get ' %s ' " %
( which , expected_failure , res ) )
2007-12-25 05:49:35 +00:00
def shouldFail2 ( self , expected_failure , which , substring ,
2008-04-15 18:11:29 +00:00
response_substring ,
2007-12-25 05:49:35 +00:00
callable , * args , * * kwargs ) :
assert substring is None or isinstance ( substring , str )
2008-04-15 18:11:29 +00:00
assert response_substring is None or isinstance ( response_substring , str )
2007-12-25 05:49:35 +00:00
d = defer . maybeDeferred ( callable , * args , * * kwargs )
def done ( res ) :
2007-12-25 08:56:04 +00:00
if isinstance ( res , failure . Failure ) :
2007-12-25 05:49:35 +00:00
res . trap ( expected_failure )
if substring :
self . failUnless ( substring in str ( res ) ,
2008-06-03 07:03:16 +00:00
" %s : substring ' %s ' not in ' %s ' "
% ( which , substring , str ( res ) ) )
2008-04-15 18:11:29 +00:00
if response_substring :
self . failUnless ( response_substring in res . value . response ,
2008-06-03 07:03:16 +00:00
" %s : response substring ' %s ' not in ' %s ' "
% ( which ,
response_substring , res . value . response ) )
2007-12-25 05:49:35 +00:00
else :
self . fail ( " %s was supposed to raise %s , not get ' %s ' " %
( which , expected_failure , res ) )
d . addBoth ( done )
return d
2007-07-07 02:43:55 +00:00
def should404 ( self , res , which ) :
if isinstance ( res , failure . Failure ) :
res . trap ( error . Error )
self . failUnlessEqual ( res . value . status , " 404 " )
else :
2007-07-07 07:16:36 +00:00
self . fail ( " %s was supposed to Error(404), not get ' %s ' " %
( which , res ) )
2007-07-07 02:43:55 +00:00
2007-12-25 05:49:35 +00:00
2009-03-03 23:56:20 +00:00
class Web ( WebMixin , WebErrorMixin , testutil . StallMixin , unittest . TestCase ) :
2007-07-25 03:16:21 +00:00
def test_create ( self ) :
2007-07-07 02:43:55 +00:00
pass
2007-07-25 03:16:21 +00:00
def test_welcome ( self ) :
2007-07-08 06:22:52 +00:00
d = self . GET ( " / " )
def _check ( res ) :
self . failUnless ( ' Welcome To AllMyData ' in res )
self . failUnless ( ' Tahoe ' in res )
2007-08-22 21:54:34 +00:00
self . s . basedir = ' web/test_welcome '
fileutil . make_dirs ( " web/test_welcome " )
2007-12-17 23:39:54 +00:00
fileutil . make_dirs ( " web/test_welcome/private " )
2007-08-22 21:54:34 +00:00
return self . GET ( " / " )
2007-07-08 06:22:52 +00:00
d . addCallback ( _check )
2007-07-07 02:43:55 +00:00
return d
2007-08-28 00:26:39 +00:00
def test_provisioning ( self ) :
d = self . GET ( " /provisioning/ " )
def _check ( res ) :
self . failUnless ( ' Tahoe Provisioning Tool ' in res )
fields = { ' filled ' : True ,
" num_users " : int ( 50e3 ) ,
" files_per_user " : 1000 ,
" space_per_user " : int ( 1e9 ) ,
" sharing_ratio " : 1.0 ,
2007-09-07 21:45:43 +00:00
" encoding_parameters " : " 3-of-10-5 " ,
2007-08-28 00:26:39 +00:00
" num_servers " : 30 ,
" ownership_mode " : " A " ,
" download_rate " : 100 ,
" upload_rate " : 10 ,
" delete_rate " : 10 ,
" lease_timer " : 7 ,
}
return self . POST ( " /provisioning/ " , * * fields )
d . addCallback ( _check )
def _check2 ( res ) :
self . failUnless ( ' Tahoe Provisioning Tool ' in res )
self . failUnless ( " Share space consumed: 167.01TB " in res )
2007-09-17 08:38:54 +00:00
fields = { ' filled ' : True ,
" num_users " : int ( 50e6 ) ,
" files_per_user " : 1000 ,
" space_per_user " : int ( 5e9 ) ,
" sharing_ratio " : 1.0 ,
" encoding_parameters " : " 25-of-100-50 " ,
" num_servers " : 30000 ,
" ownership_mode " : " E " ,
" drive_failure_model " : " U " ,
" drive_size " : 1000 ,
" download_rate " : 1000 ,
" upload_rate " : 100 ,
" delete_rate " : 100 ,
" lease_timer " : 7 ,
}
return self . POST ( " /provisioning/ " , * * fields )
2007-08-28 00:26:39 +00:00
d . addCallback ( _check2 )
2007-09-17 08:38:54 +00:00
def _check3 ( res ) :
self . failUnless ( " Share space consumed: huge! " in res )
fields = { ' filled ' : True }
return self . POST ( " /provisioning/ " , * * fields )
d . addCallback ( _check3 )
def _check4 ( res ) :
self . failUnless ( " Share space consumed: " in res )
d . addCallback ( _check4 )
2007-08-28 00:26:39 +00:00
return d
2009-02-15 23:24:51 +00:00
def test_reliability_tool ( self ) :
try :
from allmydata import reliability
_hush_pyflakes = reliability
except :
2009-02-19 08:44:35 +00:00
raise unittest . SkipTest ( " reliability tool requires NumPy " )
2009-02-15 23:24:51 +00:00
d = self . GET ( " /reliability/ " )
def _check ( res ) :
self . failUnless ( ' Tahoe Reliability Tool ' in res )
fields = { ' drive_lifetime ' : " 8Y " ,
" k " : " 3 " ,
" R " : " 7 " ,
" N " : " 10 " ,
" delta " : " 100000 " ,
" check_period " : " 1M " ,
" report_period " : " 3M " ,
" report_span " : " 5Y " ,
}
return self . POST ( " /reliability/ " , * * fields )
d . addCallback ( _check )
def _check2 ( res ) :
self . failUnless ( ' Tahoe Reliability Tool ' in res )
r = r ' Probability of loss \ (no maintenance \ ): \ s+<span>0.033591 '
self . failUnless ( re . search ( r , res ) , res )
d . addCallback ( _check2 )
return d
2008-02-13 20:57:39 +00:00
def test_status ( self ) :
2008-04-17 20:02:22 +00:00
dl_num = self . s . list_all_download_statuses ( ) [ 0 ] . get_counter ( )
ul_num = self . s . list_all_upload_statuses ( ) [ 0 ] . get_counter ( )
mu_num = self . s . list_all_mapupdate_statuses ( ) [ 0 ] . get_counter ( )
pub_num = self . s . list_all_publish_statuses ( ) [ 0 ] . get_counter ( )
ret_num = self . s . list_all_retrieve_statuses ( ) [ 0 ] . get_counter ( )
2008-03-01 05:19:03 +00:00
d = self . GET ( " /status " , followRedirect = True )
2008-02-13 20:57:39 +00:00
def _check ( res ) :
2008-03-04 08:07:44 +00:00
self . failUnless ( ' Upload and Download Status ' in res , res )
self . failUnless ( ' " down- %d " ' % dl_num in res , res )
self . failUnless ( ' " up- %d " ' % ul_num in res , res )
2008-04-17 20:02:22 +00:00
self . failUnless ( ' " mapupdate- %d " ' % mu_num in res , res )
self . failUnless ( ' " publish- %d " ' % pub_num in res , res )
self . failUnless ( ' " retrieve- %d " ' % ret_num in res , res )
2008-02-13 20:57:39 +00:00
d . addCallback ( _check )
2008-07-26 00:41:10 +00:00
d . addCallback ( lambda res : self . GET ( " /status/?t=json " ) )
def _check_json ( res ) :
data = simplejson . loads ( res )
self . failUnless ( isinstance ( data , dict ) )
active = data [ " active " ]
# TODO: test more. We need a way to fake an active operation
# here.
d . addCallback ( _check_json )
2008-03-01 06:03:00 +00:00
d . addCallback ( lambda res : self . GET ( " /status/down- %d " % dl_num ) )
def _check_dl ( res ) :
2008-03-04 08:07:44 +00:00
self . failUnless ( " File Download Status " in res , res )
2008-03-01 06:03:00 +00:00
d . addCallback ( _check_dl )
d . addCallback ( lambda res : self . GET ( " /status/up- %d " % ul_num ) )
def _check_ul ( res ) :
2008-03-04 08:07:44 +00:00
self . failUnless ( " File Upload Status " in res , res )
2008-03-01 06:03:00 +00:00
d . addCallback ( _check_ul )
2008-04-17 20:02:22 +00:00
d . addCallback ( lambda res : self . GET ( " /status/mapupdate- %d " % mu_num ) )
def _check_mapupdate ( res ) :
self . failUnless ( " Mutable File Servermap Update Status " in res , res )
d . addCallback ( _check_mapupdate )
d . addCallback ( lambda res : self . GET ( " /status/publish- %d " % pub_num ) )
def _check_publish ( res ) :
self . failUnless ( " Mutable File Publish Status " in res , res )
d . addCallback ( _check_publish )
d . addCallback ( lambda res : self . GET ( " /status/retrieve- %d " % ret_num ) )
def _check_retrieve ( res ) :
self . failUnless ( " Mutable File Retrieve Status " in res , res )
d . addCallback ( _check_retrieve )
2008-02-13 20:57:39 +00:00
return d
2008-03-04 04:56:23 +00:00
def test_status_numbers ( self ) :
2008-03-05 21:59:56 +00:00
drrm = status . DownloadResultsRendererMixin ( )
2008-03-04 04:56:23 +00:00
self . failUnlessEqual ( drrm . render_time ( None , None ) , " " )
self . failUnlessEqual ( drrm . render_time ( None , 2.5 ) , " 2.50s " )
self . failUnlessEqual ( drrm . render_time ( None , 0.25 ) , " 250ms " )
self . failUnlessEqual ( drrm . render_time ( None , 0.0021 ) , " 2.1ms " )
self . failUnlessEqual ( drrm . render_time ( None , 0.000123 ) , " 123us " )
self . failUnlessEqual ( drrm . render_rate ( None , None ) , " " )
self . failUnlessEqual ( drrm . render_rate ( None , 2500000 ) , " 2.50MBps " )
self . failUnlessEqual ( drrm . render_rate ( None , 30100 ) , " 30.1kBps " )
self . failUnlessEqual ( drrm . render_rate ( None , 123 ) , " 123Bps " )
2008-03-05 21:59:56 +00:00
urrm = status . UploadResultsRendererMixin ( )
2008-03-04 04:56:23 +00:00
self . failUnlessEqual ( urrm . render_time ( None , None ) , " " )
self . failUnlessEqual ( urrm . render_time ( None , 2.5 ) , " 2.50s " )
self . failUnlessEqual ( urrm . render_time ( None , 0.25 ) , " 250ms " )
self . failUnlessEqual ( urrm . render_time ( None , 0.0021 ) , " 2.1ms " )
self . failUnlessEqual ( urrm . render_time ( None , 0.000123 ) , " 123us " )
self . failUnlessEqual ( urrm . render_rate ( None , None ) , " " )
self . failUnlessEqual ( urrm . render_rate ( None , 2500000 ) , " 2.50MBps " )
self . failUnlessEqual ( urrm . render_rate ( None , 30100 ) , " 30.1kBps " )
self . failUnlessEqual ( urrm . render_rate ( None , 123 ) , " 123Bps " )
2007-07-25 03:16:21 +00:00
def test_GET_FILEURL ( self ) :
2007-12-03 21:52:42 +00:00
d = self . GET ( self . public_url + " /foo/bar.txt " )
2007-07-07 02:43:55 +00:00
d . addCallback ( self . failUnlessIsBarDotTxt )
2007-12-12 01:04:44 +00:00
return d
2008-10-28 20:41:04 +00:00
def test_GET_FILEURL_range ( self ) :
headers = { " range " : " bytes=1-10 " }
d = self . GET ( self . public_url + " /foo/bar.txt " , headers = headers ,
return_response = True )
def _got ( ( res , status , headers ) ) :
self . failUnlessEqual ( int ( status ) , 206 )
self . failUnless ( headers . has_key ( " content-range " ) )
self . failUnlessEqual ( headers [ " content-range " ] [ 0 ] ,
" bytes 1-10/ %d " % len ( self . BAR_CONTENTS ) )
self . failUnlessEqual ( res , self . BAR_CONTENTS [ 1 : 11 ] )
d . addCallback ( _got )
return d
2008-11-18 14:41:35 +00:00
def test_GET_FILEURL_partial_range ( self ) :
headers = { " range " : " bytes=5- " }
length = len ( self . BAR_CONTENTS )
d = self . GET ( self . public_url + " /foo/bar.txt " , headers = headers ,
return_response = True )
def _got ( ( res , status , headers ) ) :
self . failUnlessEqual ( int ( status ) , 206 )
self . failUnless ( headers . has_key ( " content-range " ) )
self . failUnlessEqual ( headers [ " content-range " ] [ 0 ] ,
" bytes 5- %d / %d " % ( length - 1 , length ) )
self . failUnlessEqual ( res , self . BAR_CONTENTS [ 5 : ] )
d . addCallback ( _got )
return d
2008-10-28 20:41:04 +00:00
def test_HEAD_FILEURL_range ( self ) :
headers = { " range " : " bytes=1-10 " }
d = self . HEAD ( self . public_url + " /foo/bar.txt " , headers = headers ,
return_response = True )
def _got ( ( res , status , headers ) ) :
self . failUnlessEqual ( res , " " )
self . failUnlessEqual ( int ( status ) , 206 )
self . failUnless ( headers . has_key ( " content-range " ) )
self . failUnlessEqual ( headers [ " content-range " ] [ 0 ] ,
" bytes 1-10/ %d " % len ( self . BAR_CONTENTS ) )
d . addCallback ( _got )
return d
2008-11-18 14:41:35 +00:00
def test_HEAD_FILEURL_partial_range ( self ) :
headers = { " range " : " bytes=5- " }
length = len ( self . BAR_CONTENTS )
d = self . HEAD ( self . public_url + " /foo/bar.txt " , headers = headers ,
return_response = True )
def _got ( ( res , status , headers ) ) :
self . failUnlessEqual ( int ( status ) , 206 )
self . failUnless ( headers . has_key ( " content-range " ) )
self . failUnlessEqual ( headers [ " content-range " ] [ 0 ] ,
" bytes 5- %d / %d " % ( length - 1 , length ) )
d . addCallback ( _got )
return d
2008-10-28 20:41:04 +00:00
def test_GET_FILEURL_range_bad ( self ) :
headers = { " range " : " BOGUS=fizbop-quarnak " }
d = self . shouldFail2 ( error . Error , " test_GET_FILEURL_range_bad " ,
" 400 Bad Request " ,
" Syntactically invalid http range header " ,
self . GET , self . public_url + " /foo/bar.txt " ,
headers = headers )
return d
2008-05-20 18:47:43 +00:00
def test_HEAD_FILEURL ( self ) :
2008-10-28 20:41:04 +00:00
d = self . HEAD ( self . public_url + " /foo/bar.txt " , return_response = True )
def _got ( ( res , status , headers ) ) :
self . failUnlessEqual ( res , " " )
2008-05-20 18:47:43 +00:00
self . failUnlessEqual ( headers [ " content-length " ] [ 0 ] ,
str ( len ( self . BAR_CONTENTS ) ) )
self . failUnlessEqual ( headers [ " content-type " ] , [ " text/plain " ] )
d . addCallback ( _got )
return d
2008-05-14 21:32:21 +00:00
def test_GET_FILEURL_named ( self ) :
base = " /file/ %s " % urllib . quote ( self . _bar_txt_uri )
2008-05-20 18:13:12 +00:00
base2 = " /named/ %s " % urllib . quote ( self . _bar_txt_uri )
2008-05-14 21:32:21 +00:00
d = self . GET ( base + " /@@name=/blah.txt " )
d . addCallback ( self . failUnlessIsBarDotTxt )
d . addCallback ( lambda res : self . GET ( base + " /blah.txt " ) )
d . addCallback ( self . failUnlessIsBarDotTxt )
d . addCallback ( lambda res : self . GET ( base + " /ignore/lots/blah.txt " ) )
d . addCallback ( self . failUnlessIsBarDotTxt )
2008-05-20 18:13:12 +00:00
d . addCallback ( lambda res : self . GET ( base2 + " /@@name=/blah.txt " ) )
d . addCallback ( self . failUnlessIsBarDotTxt )
2008-07-19 01:58:57 +00:00
save_url = base + " ?save=true&filename=blah.txt "
d . addCallback ( lambda res : self . GET ( save_url ) )
d . addCallback ( self . failUnlessIsBarDotTxt ) # TODO: check headers
u_filename = u " n \u00e9 wer.txt " # n e-acute w e r . t x t
u_fn_e = urllib . quote ( u_filename . encode ( " utf-8 " ) )
u_url = base + " ?save=true&filename= " + u_fn_e
d . addCallback ( lambda res : self . GET ( u_url ) )
d . addCallback ( self . failUnlessIsBarDotTxt ) # TODO: check headers
2008-05-14 21:32:21 +00:00
return d
2008-05-20 06:28:52 +00:00
def test_PUT_FILEURL_named_bad ( self ) :
base = " /file/ %s " % urllib . quote ( self . _bar_txt_uri )
d = self . shouldFail2 ( error . Error , " test_PUT_FILEURL_named_bad " ,
" 400 Bad Request " ,
" /file can only be used with GET or HEAD " ,
self . PUT , base + " /@@name=/blah.txt " , " " )
return d
def test_GET_DIRURL_named_bad ( self ) :
base = " /file/ %s " % urllib . quote ( self . _foo_uri )
d = self . shouldFail2 ( error . Error , " test_PUT_DIRURL_named_bad " ,
" 400 Bad Request " ,
" is not a file-cap " ,
self . GET , base + " /@@name=/blah.txt " )
return d
def test_GET_slash_file_bad ( self ) :
d = self . shouldFail2 ( error . Error , " test_GET_slash_file_bad " ,
" 404 Not Found " ,
" /file must be followed by a file-cap and a name " ,
self . GET , " /file " )
return d
2008-05-20 01:38:39 +00:00
def test_GET_unhandled_URI_named ( self ) :
contents , n , newuri = self . makefile ( 12 )
2008-12-08 19:44:11 +00:00
verifier_cap = n . get_verify_cap ( ) . to_string ( )
2008-05-20 01:38:39 +00:00
base = " /file/ %s " % urllib . quote ( verifier_cap )
# client.create_node_from_uri() can't handle verify-caps
d = self . shouldFail2 ( error . Error , " GET_unhandled_URI_named " ,
" 400 Bad Request " ,
" is not a valid file- or directory- cap " ,
self . GET , base )
return d
2008-05-20 06:28:52 +00:00
def test_GET_unhandled_URI ( self ) :
contents , n , newuri = self . makefile ( 12 )
2008-12-08 19:44:11 +00:00
verifier_cap = n . get_verify_cap ( ) . to_string ( )
2008-05-20 06:28:52 +00:00
base = " /uri/ %s " % urllib . quote ( verifier_cap )
# client.create_node_from_uri() can't handle verify-caps
d = self . shouldFail2 ( error . Error , " test_GET_unhandled_URI " ,
" 400 Bad Request " ,
" is not a valid file- or directory- cap " ,
self . GET , base )
return d
2008-05-20 18:33:14 +00:00
def test_GET_FILE_URI ( self ) :
base = " /uri/ %s " % urllib . quote ( self . _bar_txt_uri )
d = self . GET ( base )
d . addCallback ( self . failUnlessIsBarDotTxt )
return d
def test_GET_FILE_URI_badchild ( self ) :
base = " /uri/ %s /boguschild " % urllib . quote ( self . _bar_txt_uri )
errmsg = " Files have no children, certainly not named ' boguschild ' "
d = self . shouldFail2 ( error . Error , " test_GET_FILE_URI_badchild " ,
" 400 Bad Request " , errmsg ,
self . GET , base )
return d
def test_PUT_FILE_URI_badchild ( self ) :
base = " /uri/ %s /boguschild " % urllib . quote ( self . _bar_txt_uri )
errmsg = " Cannot create directory ' boguschild ' , because its parent is a file, not a directory "
d = self . shouldFail2 ( error . Error , " test_GET_FILE_URI_badchild " ,
" 400 Bad Request " , errmsg ,
self . PUT , base , " " )
return d
2007-12-12 01:04:44 +00:00
def test_GET_FILEURL_save ( self ) :
2008-05-19 19:56:02 +00:00
d = self . GET ( self . public_url + " /foo/bar.txt?filename=bar.txt&save=true " )
2007-12-12 01:04:44 +00:00
# TODO: look at the headers, expect a Content-Disposition: attachment
# header.
d . addCallback ( self . failUnlessIsBarDotTxt )
2007-07-07 02:43:55 +00:00
return d
2007-07-25 03:16:21 +00:00
def test_GET_FILEURL_missing ( self ) :
2007-12-03 21:52:42 +00:00
d = self . GET ( self . public_url + " /foo/missing " )
2007-07-07 02:43:55 +00:00
d . addBoth ( self . should404 , " test_GET_FILEURL_missing " )
return d
2007-07-25 03:16:21 +00:00
def test_PUT_NEWFILEURL ( self ) :
2007-12-03 21:52:42 +00:00
d = self . PUT ( self . public_url + " /foo/new.txt " , self . NEWFILE_CONTENTS )
2007-12-05 06:01:37 +00:00
# TODO: we lose the response code, so we can't check this
#self.failUnlessEqual(responsecode, 201)
2008-02-14 22:45:56 +00:00
d . addCallback ( self . failUnlessURIMatchesChild , self . _foo_node , u " new.txt " )
2007-12-05 06:01:37 +00:00
d . addCallback ( lambda res :
2008-02-14 22:45:56 +00:00
self . failUnlessChildContentsAre ( self . _foo_node , u " new.txt " ,
2007-12-05 06:01:37 +00:00
self . NEWFILE_CONTENTS ) )
2007-07-07 02:43:55 +00:00
return d
2009-04-08 02:13:40 +00:00
def test_PUT_NEWFILEURL_not_mutable ( self ) :
d = self . PUT ( self . public_url + " /foo/new.txt?mutable=false " ,
self . NEWFILE_CONTENTS )
# TODO: we lose the response code, so we can't check this
#self.failUnlessEqual(responsecode, 201)
d . addCallback ( self . failUnlessURIMatchesChild , self . _foo_node , u " new.txt " )
d . addCallback ( lambda res :
self . failUnlessChildContentsAre ( self . _foo_node , u " new.txt " ,
self . NEWFILE_CONTENTS ) )
return d
2008-10-28 20:41:04 +00:00
def test_PUT_NEWFILEURL_range_bad ( self ) :
headers = { " content-range " : " bytes 1-10/ %d " % len ( self . NEWFILE_CONTENTS ) }
target = self . public_url + " /foo/new.txt "
d = self . shouldFail2 ( error . Error , " test_PUT_NEWFILEURL_range_bad " ,
" 501 Not Implemented " ,
" Content-Range in PUT not yet supported " ,
# (and certainly not for immutable files)
self . PUT , target , self . NEWFILE_CONTENTS [ 1 : 11 ] ,
headers = headers )
d . addCallback ( lambda res :
self . failIfNodeHasChild ( self . _foo_node , u " new.txt " ) )
return d
2008-05-20 19:36:02 +00:00
def test_PUT_NEWFILEURL_mutable ( self ) :
d = self . PUT ( self . public_url + " /foo/new.txt?mutable=true " ,
self . NEWFILE_CONTENTS )
# TODO: we lose the response code, so we can't check this
#self.failUnlessEqual(responsecode, 201)
def _check_uri ( res ) :
u = uri . from_string_mutable_filenode ( res )
self . failUnless ( u . is_mutable ( ) )
self . failIf ( u . is_readonly ( ) )
return res
d . addCallback ( _check_uri )
d . addCallback ( self . failUnlessURIMatchesChild , self . _foo_node , u " new.txt " )
d . addCallback ( lambda res :
self . failUnlessMutableChildContentsAre ( self . _foo_node ,
u " new.txt " ,
self . NEWFILE_CONTENTS ) )
return d
2008-06-03 07:03:16 +00:00
def test_PUT_NEWFILEURL_mutable_toobig ( self ) :
d = self . shouldFail2 ( error . Error , " test_PUT_NEWFILEURL_mutable_toobig " ,
" 413 Request Entity Too Large " ,
" SDMF is limited to one segment, and 10001 > 10000 " ,
self . PUT ,
self . public_url + " /foo/new.txt?mutable=true " ,
" b " * ( self . s . MUTABLE_SIZELIMIT + 1 ) )
return d
2007-08-15 20:22:23 +00:00
def test_PUT_NEWFILEURL_replace ( self ) :
2007-12-03 21:52:42 +00:00
d = self . PUT ( self . public_url + " /foo/bar.txt " , self . NEWFILE_CONTENTS )
2007-12-05 06:01:37 +00:00
# TODO: we lose the response code, so we can't check this
#self.failUnlessEqual(responsecode, 200)
2008-02-14 22:45:56 +00:00
d . addCallback ( self . failUnlessURIMatchesChild , self . _foo_node , u " bar.txt " )
2007-12-05 06:01:37 +00:00
d . addCallback ( lambda res :
2008-02-14 22:45:56 +00:00
self . failUnlessChildContentsAre ( self . _foo_node , u " bar.txt " ,
2007-12-05 06:01:37 +00:00
self . NEWFILE_CONTENTS ) )
2007-08-15 20:22:23 +00:00
return d
2008-05-20 01:38:39 +00:00
def test_PUT_NEWFILEURL_bad_t ( self ) :
d = self . shouldFail2 ( error . Error , " PUT_bad_t " , " 400 Bad Request " ,
" PUT to a file: bad t=bogus " ,
self . PUT , self . public_url + " /foo/bar.txt?t=bogus " ,
" contents " )
return d
2007-08-15 20:22:23 +00:00
def test_PUT_NEWFILEURL_no_replace ( self ) :
2007-12-03 21:52:42 +00:00
d = self . PUT ( self . public_url + " /foo/bar.txt?replace=false " ,
2007-08-15 20:22:23 +00:00
self . NEWFILE_CONTENTS )
d . addBoth ( self . shouldFail , error . Error , " PUT_NEWFILEURL_no_replace " ,
" 409 Conflict " ,
" There was already a child by that name, and you asked me "
" to not replace it " )
return d
2007-07-25 03:16:21 +00:00
def test_PUT_NEWFILEURL_mkdirs ( self ) :
2007-12-03 21:52:42 +00:00
d = self . PUT ( self . public_url + " /foo/newdir/new.txt " , self . NEWFILE_CONTENTS )
2007-12-05 06:01:37 +00:00
fn = self . _foo_node
2008-02-14 22:45:56 +00:00
d . addCallback ( self . failUnlessURIMatchesChild , fn , u " newdir/new.txt " )
d . addCallback ( lambda res : self . failIfNodeHasChild ( fn , u " new.txt " ) )
d . addCallback ( lambda res : self . failUnlessNodeHasChild ( fn , u " newdir " ) )
2007-12-05 06:01:37 +00:00
d . addCallback ( lambda res :
2008-02-14 22:45:56 +00:00
self . failUnlessChildContentsAre ( fn , u " newdir/new.txt " ,
2007-12-05 06:01:37 +00:00
self . NEWFILE_CONTENTS ) )
2007-07-07 02:43:55 +00:00
return d
2007-07-25 03:16:21 +00:00
def test_PUT_NEWFILEURL_blocked ( self ) :
2007-12-03 21:52:42 +00:00
d = self . PUT ( self . public_url + " /foo/blockingfile/new.txt " ,
2007-07-07 02:43:55 +00:00
self . NEWFILE_CONTENTS )
d . addBoth ( self . shouldFail , error . Error , " PUT_NEWFILEURL_blocked " ,
2008-05-19 19:56:02 +00:00
" 409 Conflict " ,
" Unable to create directory ' blockingfile ' : a file was in the way " )
2007-07-07 02:43:55 +00:00
return d
2007-07-25 03:16:21 +00:00
def test_DELETE_FILEURL ( self ) :
2007-12-03 21:52:42 +00:00
d = self . DELETE ( self . public_url + " /foo/bar.txt " )
2007-12-05 06:01:37 +00:00
d . addCallback ( lambda res :
2008-02-14 22:45:56 +00:00
self . failIfNodeHasChild ( self . _foo_node , u " bar.txt " ) )
2007-07-07 02:43:55 +00:00
return d
2007-07-25 03:16:21 +00:00
def test_DELETE_FILEURL_missing ( self ) :
2007-12-03 21:52:42 +00:00
d = self . DELETE ( self . public_url + " /foo/missing " )
2007-07-07 07:16:36 +00:00
d . addBoth ( self . should404 , " test_DELETE_FILEURL_missing " )
return d
2007-07-25 03:16:21 +00:00
def test_DELETE_FILEURL_missing2 ( self ) :
2007-12-03 21:52:42 +00:00
d = self . DELETE ( self . public_url + " /missing/missing " )
2007-07-07 07:16:36 +00:00
d . addBoth ( self . should404 , " test_DELETE_FILEURL_missing2 " )
2007-07-07 02:43:55 +00:00
return d
2007-07-25 03:16:21 +00:00
def test_GET_FILEURL_json ( self ) :
2007-07-07 02:43:55 +00:00
# twisted.web.http.parse_qs ignores any query args without an '=', so
# I can't do "GET /path?json", I have to do "GET /path/t=json"
# instead. This may make it tricky to emulate the S3 interface
# completely.
2007-12-03 21:52:42 +00:00
d = self . GET ( self . public_url + " /foo/bar.txt?t=json " )
2007-07-08 07:17:11 +00:00
d . addCallback ( self . failUnlessIsBarJSON )
2007-07-07 02:43:55 +00:00
return d
2007-07-25 03:16:21 +00:00
def test_GET_FILEURL_json_missing ( self ) :
2007-12-03 21:52:42 +00:00
d = self . GET ( self . public_url + " /foo/missing?json " )
2007-07-07 02:43:55 +00:00
d . addBoth ( self . should404 , " test_GET_FILEURL_json_missing " )
return d
2007-07-25 03:16:21 +00:00
def test_GET_FILEURL_uri ( self ) :
2007-12-03 21:52:42 +00:00
d = self . GET ( self . public_url + " /foo/bar.txt?t=uri " )
2007-07-07 02:43:55 +00:00
def _check ( res ) :
self . failUnlessEqual ( res , self . _bar_txt_uri )
d . addCallback ( _check )
2007-07-08 05:47:18 +00:00
d . addCallback ( lambda res :
2007-12-03 21:52:42 +00:00
self . GET ( self . public_url + " /foo/bar.txt?t=readonly-uri " ) )
2007-07-08 05:47:18 +00:00
def _check2 ( res ) :
# for now, for files, uris and readonly-uris are the same
self . failUnlessEqual ( res , self . _bar_txt_uri )
d . addCallback ( _check2 )
2007-07-07 02:43:55 +00:00
return d
2007-12-25 10:48:57 +00:00
def test_GET_FILEURL_badtype ( self ) :
2009-03-03 23:46:04 +00:00
d = self . shouldHTTPError ( " GET t=bogus " , 400 , " Bad Request " ,
" bad t=bogus " ,
self . GET ,
self . public_url + " /foo/bar.txt?t=bogus " )
2007-12-25 10:48:57 +00:00
return d
2007-07-25 03:16:21 +00:00
def test_GET_FILEURL_uri_missing ( self ) :
2007-12-03 21:52:42 +00:00
d = self . GET ( self . public_url + " /foo/missing?t=uri " )
2007-07-07 02:43:55 +00:00
d . addBoth ( self . should404 , " test_GET_FILEURL_uri_missing " )
return d
2007-07-25 03:16:21 +00:00
def test_GET_DIRURL ( self ) :
2007-07-08 03:06:58 +00:00
# the addSlash means we get a redirect here
2008-06-18 02:49:40 +00:00
# from /uri/$URI/foo/ , we need ../../../ to get back to the root
ROOT = " ../../.. "
2007-12-03 21:52:42 +00:00
d = self . GET ( self . public_url + " /foo " , followRedirect = True )
2007-07-07 02:43:55 +00:00
def _check ( res ) :
2008-06-18 02:49:40 +00:00
self . failUnless ( ( ' <a href= " %s " >Return to Welcome page ' % ROOT )
in res , res )
2007-08-23 00:35:01 +00:00
# the FILE reference points to a URI, but it should end in bar.txt
2008-06-18 02:49:40 +00:00
bar_url = ( " %s /file/ %s /@@named=/bar.txt " %
( ROOT , urllib . quote ( self . _bar_txt_uri ) ) )
2009-05-26 23:25:45 +00:00
get_bar = " " . join ( [ r ' <td>FILE</td> ' ,
r ' \ s+<td> ' ,
2008-06-18 02:49:40 +00:00
r ' <a href= " %s " >bar.txt</a> ' % bar_url ,
r ' </td> ' ,
r ' \ s+<td> %d </td> ' % len ( self . BAR_CONTENTS ) ,
] )
self . failUnless ( re . search ( get_bar , res ) , res )
for line in res . split ( " \n " ) :
# find the line that contains the delete button for bar.txt
if ( " form action " in line and
' value= " delete " ' in line and
' value= " bar.txt " ' in line ) :
# the form target should use a relative URL
foo_url = urllib . quote ( " %s /uri/ %s / " % ( ROOT , self . _foo_uri ) )
self . failUnless ( ( ' action= " %s " ' % foo_url ) in line , line )
# and the when_done= should too
#done_url = urllib.quote(???)
#self.failUnless(('name="when_done" value="%s"' % done_url)
# in line, line)
break
else :
self . fail ( " unable to find delete-bar.txt line " , res )
2008-01-30 01:04:32 +00:00
# the DIR reference just points to a URI
2008-06-18 02:49:40 +00:00
sub_url = ( " %s /uri/ %s / " % ( ROOT , urllib . quote ( self . _sub_uri ) ) )
2009-05-26 23:25:45 +00:00
get_sub = ( ( r ' <td>DIR</td> ' )
+ r ' \ s+<td><a href= " %s " >sub</a></td> ' % sub_url )
2008-06-18 02:49:40 +00:00
self . failUnless ( re . search ( get_sub , res ) , res )
2007-07-07 02:43:55 +00:00
d . addCallback ( _check )
2007-07-08 05:47:18 +00:00
# look at a directory which is readonly
d . addCallback ( lambda res :
2007-12-03 21:52:42 +00:00
self . GET ( self . public_url + " /reedownlee " , followRedirect = True ) )
2007-07-08 05:47:18 +00:00
def _check2 ( res ) :
2009-05-26 23:24:14 +00:00
self . failUnless ( " (read-only) " in res , res )
2007-12-03 21:52:42 +00:00
self . failIf ( " Upload a file " in res , res )
2007-07-08 05:47:18 +00:00
d . addCallback ( _check2 )
# and at a directory that contains a readonly directory
d . addCallback ( lambda res :
2007-12-03 21:52:42 +00:00
self . GET ( self . public_url , followRedirect = True ) )
2007-07-08 05:47:18 +00:00
def _check3 ( res ) :
2009-05-26 23:25:45 +00:00
self . failUnless ( re . search ( ' <td>DIR-RO</td> '
r ' \ s+<td><a href= " [ \ . \ /]+/uri/URI % 3ADIR2-RO % 3A[^ " ]+ " >reedownlee</a></td> ' , res ) , res )
2007-07-08 05:47:18 +00:00
d . addCallback ( _check3 )
2009-03-20 23:58:09 +00:00
# and an empty directory
d . addCallback ( lambda res : self . GET ( self . public_url + " /foo/empty/ " ) )
def _check4 ( res ) :
2009-04-07 18:28:34 +00:00
self . failUnless ( " directory is empty " in res , res )
2009-05-26 23:24:14 +00:00
MKDIR_BUTTON_RE = re . compile ( ' <input type= " hidden " name= " t " value= " mkdir " />.*<legend class= " freeform-form-label " >Create a new directory in this directory</legend>.*<input type= " submit " value= " Create " /> ' , re . I )
2009-04-07 18:28:34 +00:00
self . failUnless ( MKDIR_BUTTON_RE . search ( res ) , res )
2009-03-20 23:58:09 +00:00
d . addCallback ( _check4 )
2007-07-08 05:21:20 +00:00
return d
2007-12-25 10:48:57 +00:00
def test_GET_DIRURL_badtype ( self ) :
2009-03-03 23:46:04 +00:00
d = self . shouldHTTPError ( " test_GET_DIRURL_badtype " ,
400 , " Bad Request " ,
" bad t=bogus " ,
self . GET ,
self . public_url + " /foo?t=bogus " )
2007-12-25 10:48:57 +00:00
return d
2007-07-25 03:16:21 +00:00
def test_GET_DIRURL_json ( self ) :
2007-12-03 21:52:42 +00:00
d = self . GET ( self . public_url + " /foo?t=json " )
2007-07-08 05:47:18 +00:00
d . addCallback ( self . failUnlessIsFooJSON )
return d
2008-10-22 00:03:07 +00:00
def test_POST_DIRURL_manifest_no_ophandle ( self ) :
d = self . shouldFail2 ( error . Error ,
" test_POST_DIRURL_manifest_no_ophandle " ,
" 400 Bad Request " ,
" slow operation requires ophandle= " ,
self . POST , self . public_url , t = " start-manifest " )
return d
def test_POST_DIRURL_manifest ( self ) :
2008-10-07 04:36:18 +00:00
d = defer . succeed ( None )
2008-10-22 00:03:07 +00:00
def getman ( ignored , output ) :
d = self . POST ( self . public_url + " /foo/?t=start-manifest&ophandle=125 " ,
followRedirect = True )
d . addCallback ( self . wait_for_operation , " 125 " )
d . addCallback ( self . get_operation_results , " 125 " , output )
return d
d . addCallback ( getman , None )
2008-10-07 04:36:18 +00:00
def _got_html ( manifest ) :
self . failUnless ( " Manifest of SI= " in manifest )
self . failUnless ( " <td>sub</td> " in manifest )
self . failUnless ( self . _sub_uri in manifest )
self . failUnless ( " <td>sub/baz.txt</td> " in manifest )
d . addCallback ( _got_html )
2008-10-23 22:56:58 +00:00
# both t=status and unadorned GET should be identical
d . addCallback ( lambda res : self . GET ( " /operations/125 " ) )
d . addCallback ( _got_html )
2008-10-22 00:03:07 +00:00
d . addCallback ( getman , " html " )
2008-10-07 04:36:18 +00:00
d . addCallback ( _got_html )
2008-10-22 00:03:07 +00:00
d . addCallback ( getman , " text " )
2008-10-07 04:36:18 +00:00
def _got_text ( manifest ) :
self . failUnless ( " \n sub " + self . _sub_uri + " \n " in manifest )
self . failUnless ( " \n sub/baz.txt URI:CHK: " in manifest )
d . addCallback ( _got_text )
2008-10-22 00:03:07 +00:00
d . addCallback ( getman , " JSON " )
2008-11-19 23:00:27 +00:00
def _got_json ( res ) :
data = res [ " manifest " ]
2008-10-07 04:36:18 +00:00
got = { }
for ( path_list , cap ) in data :
got [ tuple ( path_list ) ] = cap
self . failUnlessEqual ( got [ ( u " sub " , ) ] , self . _sub_uri )
self . failUnless ( ( u " sub " , u " baz.txt " ) in got )
2008-11-19 23:00:27 +00:00
self . failUnless ( " finished " in res )
self . failUnless ( " origin " in res )
self . failUnless ( " storage-index " in res )
2008-11-24 21:40:46 +00:00
self . failUnless ( " verifycaps " in res )
2008-11-19 23:00:27 +00:00
self . failUnless ( " stats " in res )
2008-10-07 04:36:18 +00:00
d . addCallback ( _got_json )
2007-07-07 02:43:55 +00:00
return d
2008-10-22 00:03:07 +00:00
def test_POST_DIRURL_deepsize_no_ophandle ( self ) :
d = self . shouldFail2 ( error . Error ,
" test_POST_DIRURL_deepsize_no_ophandle " ,
" 400 Bad Request " ,
" slow operation requires ophandle= " ,
self . POST , self . public_url , t = " start-deep-size " )
return d
def test_POST_DIRURL_deepsize ( self ) :
d = self . POST ( self . public_url + " /foo/?t=start-deep-size&ophandle=126 " ,
followRedirect = True )
d . addCallback ( self . wait_for_operation , " 126 " )
d . addCallback ( self . get_operation_results , " 126 " , " json " )
def _got_json ( data ) :
self . failUnlessEqual ( data [ " finished " ] , True )
size = data [ " size " ]
self . failUnless ( size > 1000 )
d . addCallback ( _got_json )
d . addCallback ( self . get_operation_results , " 126 " , " text " )
def _got_text ( res ) :
mo = re . search ( r ' ^size: ( \ d+)$ ' , res , re . M )
self . failUnless ( mo , res )
size = int ( mo . group ( 1 ) )
2008-10-07 05:11:47 +00:00
# with directories, the size varies.
self . failUnless ( size > 1000 )
2008-10-22 00:03:07 +00:00
d . addCallback ( _got_text )
return d
def test_POST_DIRURL_deepstats_no_ophandle ( self ) :
d = self . shouldFail2 ( error . Error ,
" test_POST_DIRURL_deepstats_no_ophandle " ,
" 400 Bad Request " ,
" slow operation requires ophandle= " ,
self . POST , self . public_url , t = " start-deep-stats " )
2008-03-27 18:33:42 +00:00
return d
2008-10-22 00:03:07 +00:00
def test_POST_DIRURL_deepstats ( self ) :
d = self . POST ( self . public_url + " /foo/?t=start-deep-stats&ophandle=127 " ,
followRedirect = True )
d . addCallback ( self . wait_for_operation , " 127 " )
d . addCallback ( self . get_operation_results , " 127 " , " json " )
def _got_json ( stats ) :
2008-05-08 20:21:14 +00:00
expected = { " count-immutable-files " : 3 ,
" count-mutable-files " : 0 ,
" count-literal-files " : 0 ,
" count-files " : 3 ,
" count-directories " : 3 ,
" size-immutable-files " : 57 ,
" size-literal-files " : 0 ,
#"size-directories": 1912, # varies
#"largest-directory": 1590,
" largest-directory-children " : 5 ,
" largest-immutable-file " : 19 ,
}
for k , v in expected . iteritems ( ) :
self . failUnlessEqual ( stats [ k ] , v ,
" stats[ %s ] was %s , not %s " %
( k , stats [ k ] , v ) )
2008-05-08 23:19:42 +00:00
self . failUnlessEqual ( stats [ " size-files-histogram " ] ,
[ [ 11 , 31 , 3 ] ] )
2008-10-22 00:03:07 +00:00
d . addCallback ( _got_json )
2008-05-08 20:21:14 +00:00
return d
2009-01-23 05:01:36 +00:00
def test_POST_DIRURL_stream_manifest ( self ) :
d = self . POST ( self . public_url + " /foo/?t=stream-manifest " )
def _check ( res ) :
self . failUnless ( res . endswith ( " \n " ) )
units = [ simplejson . loads ( t ) for t in res [ : - 1 ] . split ( " \n " ) ]
self . failUnlessEqual ( len ( units ) , 7 )
self . failUnlessEqual ( units [ - 1 ] [ " type " ] , " stats " )
first = units [ 0 ]
self . failUnlessEqual ( first [ " path " ] , [ ] )
self . failUnlessEqual ( first [ " cap " ] , self . _foo_uri )
self . failUnlessEqual ( first [ " type " ] , " directory " )
baz = [ u for u in units [ : - 1 ] if u [ " cap " ] == self . _baz_file_uri ] [ 0 ]
self . failUnlessEqual ( baz [ " path " ] , [ " sub " , " baz.txt " ] )
self . failIfEqual ( baz [ " storage-index " ] , None )
self . failIfEqual ( baz [ " verifycap " ] , None )
self . failIfEqual ( baz [ " repaircap " ] , None )
return
d . addCallback ( _check )
return d
2007-07-25 03:16:21 +00:00
def test_GET_DIRURL_uri ( self ) :
2007-12-03 21:52:42 +00:00
d = self . GET ( self . public_url + " /foo?t=uri " )
2007-07-07 02:43:55 +00:00
def _check ( res ) :
self . failUnlessEqual ( res , self . _foo_uri )
d . addCallback ( _check )
return d
2007-07-25 03:16:21 +00:00
def test_GET_DIRURL_readonly_uri ( self ) :
2007-12-03 21:52:42 +00:00
d = self . GET ( self . public_url + " /foo?t=readonly-uri " )
2007-07-07 02:43:55 +00:00
def _check ( res ) :
self . failUnlessEqual ( res , self . _foo_readonly_uri )
d . addCallback ( _check )
return d
2007-07-25 03:16:21 +00:00
def test_PUT_NEWDIRURL ( self ) :
2007-12-03 21:52:42 +00:00
d = self . PUT ( self . public_url + " /foo/newdir?t=mkdir " , " " )
2007-12-05 06:01:37 +00:00
d . addCallback ( lambda res :
2008-02-14 22:45:56 +00:00
self . failUnlessNodeHasChild ( self . _foo_node , u " newdir " ) )
d . addCallback ( lambda res : self . _foo_node . get ( u " newdir " ) )
2007-12-05 06:01:37 +00:00
d . addCallback ( self . failUnlessNodeKeysAre , [ ] )
2007-07-07 02:43:55 +00:00
return d
2008-05-19 19:56:02 +00:00
def test_PUT_NEWDIRURL_exists ( self ) :
2007-12-03 21:52:42 +00:00
d = self . PUT ( self . public_url + " /foo/sub?t=mkdir " , " " )
2007-12-05 06:01:37 +00:00
d . addCallback ( lambda res :
2008-02-14 22:45:56 +00:00
self . failUnlessNodeHasChild ( self . _foo_node , u " sub " ) )
d . addCallback ( lambda res : self . _foo_node . get ( u " sub " ) )
2008-05-19 19:56:02 +00:00
d . addCallback ( self . failUnlessNodeKeysAre , [ u " baz.txt " ] )
2007-08-15 20:22:23 +00:00
return d
2008-05-19 19:56:02 +00:00
def test_PUT_NEWDIRURL_blocked ( self ) :
d = self . shouldFail2 ( error . Error , " PUT_NEWDIRURL_blocked " ,
" 409 Conflict " , " Unable to create directory ' bar.txt ' : a file was in the way " ,
self . PUT ,
self . public_url + " /foo/bar.txt/sub?t=mkdir " , " " )
2007-12-05 06:01:37 +00:00
d . addCallback ( lambda res :
2008-02-14 22:45:56 +00:00
self . failUnlessNodeHasChild ( self . _foo_node , u " sub " ) )
d . addCallback ( lambda res : self . _foo_node . get ( u " sub " ) )
d . addCallback ( self . failUnlessNodeKeysAre , [ u " baz.txt " ] )
2007-08-15 20:22:23 +00:00
return d
2008-03-18 20:11:08 +00:00
def test_PUT_NEWDIRURL_mkdir_p ( self ) :
d = defer . succeed ( None )
d . addCallback ( lambda res : self . POST ( self . public_url + " /foo " , t = ' mkdir ' , name = ' mkp ' ) )
d . addCallback ( lambda res : self . failUnlessNodeHasChild ( self . _foo_node , u " mkp " ) )
d . addCallback ( lambda res : self . _foo_node . get ( u " mkp " ) )
def mkdir_p ( mkpnode ) :
url = ' /uri/ %s ?t=mkdir-p&path=/sub1/sub2 ' % urllib . quote ( mkpnode . get_uri ( ) )
d = self . POST ( url )
def made_subsub ( ssuri ) :
d = self . _foo_node . get_child_at_path ( u " mkp/sub1/sub2 " )
d . addCallback ( lambda ssnode : self . failUnlessEqual ( ssnode . get_uri ( ) , ssuri ) )
d = self . POST ( url )
d . addCallback ( lambda uri2 : self . failUnlessEqual ( uri2 , ssuri ) )
return d
d . addCallback ( made_subsub )
return d
d . addCallback ( mkdir_p )
return d
2007-07-25 03:16:21 +00:00
def test_PUT_NEWDIRURL_mkdirs ( self ) :
2007-12-03 21:52:42 +00:00
d = self . PUT ( self . public_url + " /foo/subdir/newdir?t=mkdir " , " " )
2007-12-05 06:01:37 +00:00
d . addCallback ( lambda res :
2008-02-14 22:45:56 +00:00
self . failIfNodeHasChild ( self . _foo_node , u " newdir " ) )
2007-12-05 06:01:37 +00:00
d . addCallback ( lambda res :
2008-02-14 22:45:56 +00:00
self . failUnlessNodeHasChild ( self . _foo_node , u " subdir " ) )
2007-12-05 06:01:37 +00:00
d . addCallback ( lambda res :
2008-02-14 22:45:56 +00:00
self . _foo_node . get_child_at_path ( u " subdir/newdir " ) )
2007-12-05 06:01:37 +00:00
d . addCallback ( self . failUnlessNodeKeysAre , [ ] )
2007-07-07 02:43:55 +00:00
return d
2007-07-25 03:16:21 +00:00
def test_DELETE_DIRURL ( self ) :
2007-12-03 21:52:42 +00:00
d = self . DELETE ( self . public_url + " /foo " )
2007-12-05 06:01:37 +00:00
d . addCallback ( lambda res :
2008-02-14 22:45:56 +00:00
self . failIfNodeHasChild ( self . public_root , u " foo " ) )
2007-07-07 02:43:55 +00:00
return d
2007-07-25 03:16:21 +00:00
def test_DELETE_DIRURL_missing ( self ) :
2007-12-03 21:52:42 +00:00
d = self . DELETE ( self . public_url + " /foo/missing " )
2007-07-07 07:16:36 +00:00
d . addBoth ( self . should404 , " test_DELETE_DIRURL_missing " )
2007-12-05 06:01:37 +00:00
d . addCallback ( lambda res :
2008-02-14 22:45:56 +00:00
self . failUnlessNodeHasChild ( self . public_root , u " foo " ) )
2007-07-07 07:16:36 +00:00
return d
2007-07-25 03:16:21 +00:00
def test_DELETE_DIRURL_missing2 ( self ) :
2007-12-03 21:52:42 +00:00
d = self . DELETE ( self . public_url + " /missing " )
2007-07-07 07:16:36 +00:00
d . addBoth ( self . should404 , " test_DELETE_DIRURL_missing2 " )
2007-07-07 02:43:55 +00:00
return d
2007-07-07 17:34:05 +00:00
def dump_root ( self ) :
print " NODEWALK "
2007-12-05 06:11:00 +00:00
w = webish . DirnodeWalkerMixin ( )
def visitor ( childpath , childnode , metadata ) :
print childpath
d = w . walk ( self . public_root , visitor )
return d
2007-07-07 17:34:05 +00:00
2007-12-05 06:01:37 +00:00
def failUnlessNodeKeysAre ( self , node , expected_keys ) :
2008-02-14 22:45:56 +00:00
for k in expected_keys :
assert isinstance ( k , unicode )
2007-12-05 06:01:37 +00:00
d = node . list ( )
def _check ( children ) :
self . failUnlessEqual ( sorted ( children . keys ( ) ) , sorted ( expected_keys ) )
d . addCallback ( _check )
return d
def failUnlessNodeHasChild ( self , node , name ) :
2008-02-14 22:45:56 +00:00
assert isinstance ( name , unicode )
2007-12-05 06:01:37 +00:00
d = node . list ( )
def _check ( children ) :
self . failUnless ( name in children )
d . addCallback ( _check )
return d
def failIfNodeHasChild ( self , node , name ) :
2008-02-14 22:45:56 +00:00
assert isinstance ( name , unicode )
2007-12-05 06:01:37 +00:00
d = node . list ( )
def _check ( children ) :
self . failIf ( name in children )
d . addCallback ( _check )
return d
def failUnlessChildContentsAre ( self , node , name , expected_contents ) :
2008-02-14 22:45:56 +00:00
assert isinstance ( name , unicode )
2007-12-05 06:01:37 +00:00
d = node . get_child_at_path ( name )
d . addCallback ( lambda node : node . download_to_data ( ) )
def _check ( contents ) :
self . failUnlessEqual ( contents , expected_contents )
d . addCallback ( _check )
return d
2008-04-18 00:51:38 +00:00
def failUnlessMutableChildContentsAre ( self , node , name , expected_contents ) :
assert isinstance ( name , unicode )
d = node . get_child_at_path ( name )
d . addCallback ( lambda node : node . download_best_version ( ) )
def _check ( contents ) :
self . failUnlessEqual ( contents , expected_contents )
d . addCallback ( _check )
return d
2007-12-05 06:01:37 +00:00
def failUnlessChildURIIs ( self , node , name , expected_uri ) :
2008-02-14 22:45:56 +00:00
assert isinstance ( name , unicode )
2007-12-05 06:01:37 +00:00
d = node . get_child_at_path ( name )
def _check ( child ) :
self . failUnlessEqual ( child . get_uri ( ) , expected_uri . strip ( ) )
d . addCallback ( _check )
return d
def failUnlessURIMatchesChild ( self , got_uri , node , name ) :
2008-02-14 22:45:56 +00:00
assert isinstance ( name , unicode )
2007-12-05 06:01:37 +00:00
d = node . get_child_at_path ( name )
def _check ( child ) :
self . failUnlessEqual ( got_uri . strip ( ) , child . get_uri ( ) )
d . addCallback ( _check )
return d
2007-12-07 00:17:02 +00:00
def failUnlessCHKURIHasContents ( self , got_uri , contents ) :
self . failUnless ( FakeCHKFileNode . all_contents [ got_uri ] == contents )
2007-07-25 03:16:21 +00:00
def test_POST_upload ( self ) :
2007-12-03 21:52:42 +00:00
d = self . POST ( self . public_url + " /foo " , t = " upload " ,
2007-07-08 03:06:58 +00:00
file = ( " new.txt " , self . NEWFILE_CONTENTS ) )
2007-12-05 06:01:37 +00:00
fn = self . _foo_node
2008-02-14 22:45:56 +00:00
d . addCallback ( self . failUnlessURIMatchesChild , fn , u " new.txt " )
d . addCallback ( lambda res :
self . failUnlessChildContentsAre ( fn , u " new.txt " ,
self . NEWFILE_CONTENTS ) )
return d
def test_POST_upload_unicode ( self ) :
filename = u " n \u00e9 wer.txt " # n e-acute w e r . t x t
d = self . POST ( self . public_url + " /foo " , t = " upload " ,
file = ( filename , self . NEWFILE_CONTENTS ) )
fn = self . _foo_node
d . addCallback ( self . failUnlessURIMatchesChild , fn , filename )
2007-12-05 06:01:37 +00:00
d . addCallback ( lambda res :
2008-02-14 22:45:56 +00:00
self . failUnlessChildContentsAre ( fn , filename ,
2007-12-05 06:01:37 +00:00
self . NEWFILE_CONTENTS ) )
2008-06-04 00:09:39 +00:00
target_url = self . public_url + " /foo/ " + filename . encode ( " utf-8 " )
d . addCallback ( lambda res : self . GET ( target_url ) )
d . addCallback ( lambda contents : self . failUnlessEqual ( contents ,
self . NEWFILE_CONTENTS ,
contents ) )
return d
def test_POST_upload_unicode_named ( self ) :
filename = u " n \u00e9 wer.txt " # n e-acute w e r . t x t
d = self . POST ( self . public_url + " /foo " , t = " upload " ,
name = filename ,
file = ( " overridden " , self . NEWFILE_CONTENTS ) )
fn = self . _foo_node
d . addCallback ( self . failUnlessURIMatchesChild , fn , filename )
d . addCallback ( lambda res :
self . failUnlessChildContentsAre ( fn , filename ,
self . NEWFILE_CONTENTS ) )
target_url = self . public_url + " /foo/ " + filename . encode ( " utf-8 " )
2008-02-14 22:45:56 +00:00
d . addCallback ( lambda res : self . GET ( target_url ) )
d . addCallback ( lambda contents : self . failUnlessEqual ( contents ,
self . NEWFILE_CONTENTS ,
contents ) )
2007-07-07 02:43:55 +00:00
return d
2007-12-07 00:17:02 +00:00
def test_POST_upload_no_link ( self ) :
2007-12-25 00:55:38 +00:00
d = self . POST ( " /uri " , t = " upload " ,
2007-12-07 00:17:02 +00:00
file = ( " new.txt " , self . NEWFILE_CONTENTS ) )
2008-02-06 06:01:37 +00:00
def _check_upload_results ( page ) :
# this should be a page which describes the results of the upload
# that just finished.
self . failUnless ( " Upload Results: " in page )
self . failUnless ( " URI: " in page )
uri_re = re . compile ( " URI: <tt><span>(.*)</span> " )
mo = uri_re . search ( page )
self . failUnless ( mo , page )
new_uri = mo . group ( 1 )
return new_uri
d . addCallback ( _check_upload_results )
2007-12-07 00:17:02 +00:00
d . addCallback ( self . failUnlessCHKURIHasContents , self . NEWFILE_CONTENTS )
return d
def test_POST_upload_no_link_whendone ( self ) :
2007-12-25 00:55:38 +00:00
d = self . POST ( " /uri " , t = " upload " , when_done = " / " ,
2007-12-07 00:17:02 +00:00
file = ( " new.txt " , self . NEWFILE_CONTENTS ) )
d . addBoth ( self . shouldRedirect , " / " )
return d
2008-05-20 06:28:52 +00:00
def shouldRedirect2 ( self , which , checker , callable , * args , * * kwargs ) :
d = defer . maybeDeferred ( callable , * args , * * kwargs )
def done ( res ) :
if isinstance ( res , failure . Failure ) :
res . trap ( error . PageRedirect )
statuscode = res . value . status
target = res . value . location
return checker ( statuscode , target )
self . fail ( " %s : callable was supposed to redirect, not return ' %s ' "
% ( which , res ) )
d . addBoth ( done )
return d
def test_POST_upload_no_link_whendone_results ( self ) :
def check ( statuscode , target ) :
self . failUnlessEqual ( statuscode , str ( http . FOUND ) )
self . failUnless ( target . startswith ( self . webish_url ) , target )
return client . getPage ( target , method = " GET " )
d = self . shouldRedirect2 ( " test_POST_upload_no_link_whendone_results " ,
check ,
self . POST , " /uri " , t = " upload " ,
when_done = " /uri/ %(uri)s " ,
file = ( " new.txt " , self . NEWFILE_CONTENTS ) )
d . addCallback ( lambda res :
self . failUnlessEqual ( res , self . NEWFILE_CONTENTS ) )
return d
2008-02-06 05:10:22 +00:00
def test_POST_upload_no_link_mutable ( self ) :
d = self . POST ( " /uri " , t = " upload " , mutable = " true " ,
file = ( " new.txt " , self . NEWFILE_CONTENTS ) )
2008-02-06 05:37:51 +00:00
def _check ( new_uri ) :
new_uri = new_uri . strip ( )
2008-05-20 01:38:39 +00:00
self . new_uri = new_uri
2008-02-06 05:37:51 +00:00
u = IURI ( new_uri )
2008-02-06 05:10:22 +00:00
self . failUnless ( IMutableFileURI . providedBy ( u ) )
2008-02-06 05:18:02 +00:00
self . failUnless ( u . storage_index in FakeMutableFileNode . all_contents )
2008-02-06 05:37:51 +00:00
n = self . s . create_node_from_uri ( new_uri )
2008-04-18 00:51:38 +00:00
return n . download_best_version ( )
2008-02-06 05:10:22 +00:00
d . addCallback ( _check )
def _check2 ( data ) :
self . failUnlessEqual ( data , self . NEWFILE_CONTENTS )
2008-05-20 01:38:39 +00:00
return self . GET ( " /uri/ %s " % urllib . quote ( self . new_uri ) )
2008-02-06 05:10:22 +00:00
d . addCallback ( _check2 )
2008-05-20 01:38:39 +00:00
def _check3 ( data ) :
self . failUnlessEqual ( data , self . NEWFILE_CONTENTS )
return self . GET ( " /file/ %s " % urllib . quote ( self . new_uri ) )
d . addCallback ( _check3 )
def _check4 ( data ) :
self . failUnlessEqual ( data , self . NEWFILE_CONTENTS )
d . addCallback ( _check4 )
2008-02-06 05:10:22 +00:00
return d
2008-06-03 07:03:16 +00:00
def test_POST_upload_no_link_mutable_toobig ( self ) :
d = self . shouldFail2 ( error . Error ,
" test_POST_upload_no_link_mutable_toobig " ,
" 413 Request Entity Too Large " ,
" SDMF is limited to one segment, and 10001 > 10000 " ,
self . POST ,
" /uri " , t = " upload " , mutable = " true " ,
file = ( " new.txt " ,
" b " * ( self . s . MUTABLE_SIZELIMIT + 1 ) ) )
return d
2007-12-05 06:42:54 +00:00
def test_POST_upload_mutable ( self ) :
# this creates a mutable file
d = self . POST ( self . public_url + " /foo " , t = " upload " , mutable = " true " ,
file = ( " new.txt " , self . NEWFILE_CONTENTS ) )
fn = self . _foo_node
2008-02-14 22:45:56 +00:00
d . addCallback ( self . failUnlessURIMatchesChild , fn , u " new.txt " )
2007-12-05 06:42:54 +00:00
d . addCallback ( lambda res :
2008-04-18 00:51:38 +00:00
self . failUnlessMutableChildContentsAre ( fn , u " new.txt " ,
self . NEWFILE_CONTENTS ) )
2008-02-14 22:45:56 +00:00
d . addCallback ( lambda res : self . _foo_node . get ( u " new.txt " ) )
2007-12-05 06:42:54 +00:00
def _got ( newnode ) :
self . failUnless ( IMutableFileNode . providedBy ( newnode ) )
self . failUnless ( newnode . is_mutable ( ) )
self . failIf ( newnode . is_readonly ( ) )
2008-05-20 01:37:28 +00:00
self . _mutable_node = newnode
2007-12-05 06:42:54 +00:00
self . _mutable_uri = newnode . get_uri ( )
d . addCallback ( _got )
# now upload it again and make sure that the URI doesn't change
NEWER_CONTENTS = self . NEWFILE_CONTENTS + " newer \n "
d . addCallback ( lambda res :
self . POST ( self . public_url + " /foo " , t = " upload " ,
mutable = " true " ,
file = ( " new.txt " , NEWER_CONTENTS ) ) )
2008-02-14 22:45:56 +00:00
d . addCallback ( self . failUnlessURIMatchesChild , fn , u " new.txt " )
2007-12-05 06:42:54 +00:00
d . addCallback ( lambda res :
2008-04-18 00:51:38 +00:00
self . failUnlessMutableChildContentsAre ( fn , u " new.txt " ,
NEWER_CONTENTS ) )
2008-02-14 22:45:56 +00:00
d . addCallback ( lambda res : self . _foo_node . get ( u " new.txt " ) )
2007-12-05 06:42:54 +00:00
def _got2 ( newnode ) :
self . failUnless ( IMutableFileNode . providedBy ( newnode ) )
self . failUnless ( newnode . is_mutable ( ) )
self . failIf ( newnode . is_readonly ( ) )
self . failUnlessEqual ( self . _mutable_uri , newnode . get_uri ( ) )
d . addCallback ( _got2 )
2008-05-20 01:38:39 +00:00
# upload a second time, using PUT instead of POST
NEW2_CONTENTS = NEWER_CONTENTS + " overwrite with PUT \n "
d . addCallback ( lambda res :
self . PUT ( self . public_url + " /foo/new.txt " , NEW2_CONTENTS ) )
d . addCallback ( self . failUnlessURIMatchesChild , fn , u " new.txt " )
d . addCallback ( lambda res :
self . failUnlessMutableChildContentsAre ( fn , u " new.txt " ,
NEW2_CONTENTS ) )
2007-12-05 06:57:40 +00:00
# finally list the directory, since mutable files are displayed
2008-09-18 05:00:41 +00:00
# slightly differently
2007-12-05 06:57:40 +00:00
d . addCallback ( lambda res :
2008-06-11 22:19:17 +00:00
self . GET ( self . public_url + " /foo/ " ,
2007-12-05 06:57:40 +00:00
followRedirect = True ) )
def _check_page ( res ) :
2007-12-05 07:15:13 +00:00
# TODO: assert more about the contents
2008-09-18 05:00:41 +00:00
self . failUnless ( " SSK " in res )
2007-12-15 00:52:05 +00:00
return res
2007-12-05 06:57:40 +00:00
d . addCallback ( _check_page )
2008-02-14 22:45:56 +00:00
d . addCallback ( lambda res : self . _foo_node . get ( u " new.txt " ) )
2007-12-15 00:52:05 +00:00
def _got3 ( newnode ) :
self . failUnless ( IMutableFileNode . providedBy ( newnode ) )
self . failUnless ( newnode . is_mutable ( ) )
self . failIf ( newnode . is_readonly ( ) )
self . failUnlessEqual ( self . _mutable_uri , newnode . get_uri ( ) )
d . addCallback ( _got3 )
2008-05-20 22:14:19 +00:00
# look at the JSON form of the enclosing directory
2008-05-20 01:37:28 +00:00
d . addCallback ( lambda res :
self . GET ( self . public_url + " /foo/?t=json " ,
followRedirect = True ) )
def _check_page_json ( res ) :
parsed = simplejson . loads ( res )
self . failUnlessEqual ( parsed [ 0 ] , " dirnode " )
2008-09-30 22:21:06 +00:00
children = dict ( [ ( unicode ( name ) , value )
for ( name , value )
in parsed [ 1 ] [ " children " ] . iteritems ( ) ] )
2008-05-20 01:37:28 +00:00
self . failUnless ( " new.txt " in children )
new_json = children [ " new.txt " ]
self . failUnlessEqual ( new_json [ 0 ] , " filenode " )
2008-05-20 22:40:49 +00:00
self . failUnless ( new_json [ 1 ] [ " mutable " ] )
2008-05-20 01:37:28 +00:00
self . failUnlessEqual ( new_json [ 1 ] [ " rw_uri " ] , self . _mutable_uri )
ro_uri = unicode ( self . _mutable_node . get_readonly ( ) . to_string ( ) )
self . failUnlessEqual ( new_json [ 1 ] [ " ro_uri " ] , ro_uri )
d . addCallback ( _check_page_json )
2008-05-20 22:14:19 +00:00
# and the JSON form of the file
d . addCallback ( lambda res :
self . GET ( self . public_url + " /foo/new.txt?t=json " ) )
def _check_file_json ( res ) :
parsed = simplejson . loads ( res )
self . failUnlessEqual ( parsed [ 0 ] , " filenode " )
2008-05-20 22:40:49 +00:00
self . failUnless ( parsed [ 1 ] [ " mutable " ] )
2008-05-20 22:14:19 +00:00
self . failUnlessEqual ( parsed [ 1 ] [ " rw_uri " ] , self . _mutable_uri )
ro_uri = unicode ( self . _mutable_node . get_readonly ( ) . to_string ( ) )
self . failUnlessEqual ( parsed [ 1 ] [ " ro_uri " ] , ro_uri )
d . addCallback ( _check_file_json )
2008-05-20 18:33:14 +00:00
# and look at t=uri and t=readonly-uri
d . addCallback ( lambda res :
self . GET ( self . public_url + " /foo/new.txt?t=uri " ) )
d . addCallback ( lambda res : self . failUnlessEqual ( res , self . _mutable_uri ) )
d . addCallback ( lambda res :
self . GET ( self . public_url + " /foo/new.txt?t=readonly-uri " ) )
def _check_ro_uri ( res ) :
ro_uri = unicode ( self . _mutable_node . get_readonly ( ) . to_string ( ) )
self . failUnlessEqual ( res , ro_uri )
d . addCallback ( _check_ro_uri )
2008-05-20 18:35:47 +00:00
# make sure we can get to it from /uri/URI
d . addCallback ( lambda res :
self . GET ( " /uri/ %s " % urllib . quote ( self . _mutable_uri ) ) )
d . addCallback ( lambda res :
2008-09-18 05:00:41 +00:00
self . failUnlessEqual ( res , NEW2_CONTENTS ) )
2008-05-20 18:35:47 +00:00
2008-05-20 18:47:43 +00:00
# and that HEAD computes the size correctly
d . addCallback ( lambda res :
2008-10-28 20:41:04 +00:00
self . HEAD ( self . public_url + " /foo/new.txt " ,
return_response = True ) )
def _got_headers ( ( res , status , headers ) ) :
self . failUnlessEqual ( res , " " )
2008-05-20 18:47:43 +00:00
self . failUnlessEqual ( headers [ " content-length " ] [ 0 ] ,
2008-09-18 05:00:41 +00:00
str ( len ( NEW2_CONTENTS ) ) )
2008-05-20 18:47:43 +00:00
self . failUnlessEqual ( headers [ " content-type " ] , [ " text/plain " ] )
d . addCallback ( _got_headers )
2008-06-03 07:03:16 +00:00
# make sure that size errors are displayed correctly for overwrite
d . addCallback ( lambda res :
self . shouldFail2 ( error . Error ,
" test_POST_upload_mutable-toobig " ,
" 413 Request Entity Too Large " ,
" SDMF is limited to one segment, and 10001 > 10000 " ,
self . POST ,
self . public_url + " /foo " , t = " upload " ,
mutable = " true " ,
file = ( " new.txt " ,
" b " * ( self . s . MUTABLE_SIZELIMIT + 1 ) ) ,
) )
2008-05-19 19:33:39 +00:00
d . addErrback ( self . dump_error )
2007-12-05 06:42:54 +00:00
return d
2008-06-03 07:03:16 +00:00
def test_POST_upload_mutable_toobig ( self ) :
d = self . shouldFail2 ( error . Error ,
" test_POST_upload_no_link_mutable_toobig " ,
" 413 Request Entity Too Large " ,
" SDMF is limited to one segment, and 10001 > 10000 " ,
self . POST ,
self . public_url + " /foo " ,
t = " upload " , mutable = " true " ,
file = ( " new.txt " ,
" b " * ( self . s . MUTABLE_SIZELIMIT + 1 ) ) )
return d
2008-05-19 19:33:39 +00:00
def dump_error ( self , f ) :
# if the web server returns an error code (like 400 Bad Request),
# web.client.getPage puts the HTTP response body into the .response
# attribute of the exception object that it gives back. It does not
# appear in the Failure's repr(), so the ERROR that trial displays
# will be rather terse and unhelpful. addErrback this method to the
# end of your chain to get more information out of these errors.
if f . check ( error . Error ) :
print " web.error.Error: "
print f
print f . value . response
return f
2007-08-15 22:21:38 +00:00
def test_POST_upload_replace ( self ) :
2007-12-03 21:52:42 +00:00
d = self . POST ( self . public_url + " /foo " , t = " upload " ,
2007-08-15 22:21:38 +00:00
file = ( " bar.txt " , self . NEWFILE_CONTENTS ) )
2007-12-05 06:01:37 +00:00
fn = self . _foo_node
2008-02-14 22:45:56 +00:00
d . addCallback ( self . failUnlessURIMatchesChild , fn , u " bar.txt " )
2007-12-05 06:01:37 +00:00
d . addCallback ( lambda res :
2008-02-14 22:45:56 +00:00
self . failUnlessChildContentsAre ( fn , u " bar.txt " ,
2007-12-05 06:01:37 +00:00
self . NEWFILE_CONTENTS ) )
2007-08-15 22:21:38 +00:00
return d
2007-12-25 10:48:57 +00:00
def test_POST_upload_no_replace_ok ( self ) :
d = self . POST ( self . public_url + " /foo?replace=false " , t = " upload " ,
file = ( " new.txt " , self . NEWFILE_CONTENTS ) )
d . addCallback ( lambda res : self . GET ( self . public_url + " /foo/new.txt " ) )
d . addCallback ( lambda res : self . failUnlessEqual ( res ,
self . NEWFILE_CONTENTS ) )
return d
2007-08-15 22:21:38 +00:00
def test_POST_upload_no_replace_queryarg ( self ) :
2007-12-03 21:52:42 +00:00
d = self . POST ( self . public_url + " /foo?replace=false " , t = " upload " ,
2007-08-15 22:21:38 +00:00
file = ( " bar.txt " , self . NEWFILE_CONTENTS ) )
d . addBoth ( self . shouldFail , error . Error ,
" POST_upload_no_replace_queryarg " ,
" 409 Conflict " ,
" There was already a child by that name, and you asked me "
" to not replace it " )
2007-12-03 21:52:42 +00:00
d . addCallback ( lambda res : self . GET ( self . public_url + " /foo/bar.txt " ) )
2007-08-15 22:21:38 +00:00
d . addCallback ( self . failUnlessIsBarDotTxt )
return d
def test_POST_upload_no_replace_field ( self ) :
2007-12-03 21:52:42 +00:00
d = self . POST ( self . public_url + " /foo " , t = " upload " , replace = " false " ,
2007-08-15 22:21:38 +00:00
file = ( " bar.txt " , self . NEWFILE_CONTENTS ) )
d . addBoth ( self . shouldFail , error . Error , " POST_upload_no_replace_field " ,
" 409 Conflict " ,
" There was already a child by that name, and you asked me "
" to not replace it " )
2007-12-03 21:52:42 +00:00
d . addCallback ( lambda res : self . GET ( self . public_url + " /foo/bar.txt " ) )
2007-08-15 22:21:38 +00:00
d . addCallback ( self . failUnlessIsBarDotTxt )
return d
2007-08-14 00:45:02 +00:00
def test_POST_upload_whendone ( self ) :
2007-12-03 21:52:42 +00:00
d = self . POST ( self . public_url + " /foo " , t = " upload " , when_done = " /THERE " ,
2007-08-14 00:45:02 +00:00
file = ( " new.txt " , self . NEWFILE_CONTENTS ) )
d . addBoth ( self . shouldRedirect , " /THERE " )
2007-12-05 06:01:37 +00:00
fn = self . _foo_node
d . addCallback ( lambda res :
2008-02-14 22:45:56 +00:00
self . failUnlessChildContentsAre ( fn , u " new.txt " ,
2007-12-05 06:01:37 +00:00
self . NEWFILE_CONTENTS ) )
2007-08-14 00:45:02 +00:00
return d
2007-07-25 03:16:21 +00:00
def test_POST_upload_named ( self ) :
2007-12-05 06:01:37 +00:00
fn = self . _foo_node
2007-12-03 21:52:42 +00:00
d = self . POST ( self . public_url + " /foo " , t = " upload " ,
2007-07-08 03:06:58 +00:00
name = " new.txt " , file = self . NEWFILE_CONTENTS )
2008-02-14 22:45:56 +00:00
d . addCallback ( self . failUnlessURIMatchesChild , fn , u " new.txt " )
2007-12-05 06:01:37 +00:00
d . addCallback ( lambda res :
2008-02-14 22:45:56 +00:00
self . failUnlessChildContentsAre ( fn , u " new.txt " ,
2007-12-05 06:01:37 +00:00
self . NEWFILE_CONTENTS ) )
2007-07-07 02:43:55 +00:00
return d
2007-07-25 03:16:21 +00:00
def test_POST_upload_named_badfilename ( self ) :
2007-12-03 21:52:42 +00:00
d = self . POST ( self . public_url + " /foo " , t = " upload " ,
2007-07-16 18:53:12 +00:00
name = " slashes/are/bad.txt " , file = self . NEWFILE_CONTENTS )
d . addBoth ( self . shouldFail , error . Error ,
" test_POST_upload_named_badfilename " ,
" 400 Bad Request " ,
" name= may not contain a slash " ,
)
2007-12-05 06:01:37 +00:00
# make sure that nothing was added
d . addCallback ( lambda res :
self . failUnlessNodeKeysAre ( self . _foo_node ,
2008-02-14 22:45:56 +00:00
[ u " bar.txt " , u " blockingfile " ,
u " empty " , u " n \u00fc .txt " ,
u " sub " ] ) )
2007-07-16 18:53:12 +00:00
return d
2008-05-20 18:33:14 +00:00
def test_POST_FILEURL_check ( self ) :
2008-07-16 22:42:56 +00:00
bar_url = self . public_url + " /foo/bar.txt "
d = self . POST ( bar_url , t = " check " )
2008-05-20 18:33:14 +00:00
def _check ( res ) :
2008-11-07 05:35:47 +00:00
self . failUnless ( " Healthy : " in res )
2008-05-20 18:33:14 +00:00
d . addCallback ( _check )
2008-07-16 22:42:56 +00:00
redir_url = " http://allmydata.org/TARGET "
def _check2 ( statuscode , target ) :
self . failUnlessEqual ( statuscode , str ( http . FOUND ) )
self . failUnlessEqual ( target , redir_url )
d . addCallback ( lambda res :
self . shouldRedirect2 ( " test_POST_FILEURL_check " ,
_check2 ,
self . POST , bar_url ,
t = " check " ,
when_done = redir_url ) )
d . addCallback ( lambda res :
self . POST ( bar_url , t = " check " , return_to = redir_url ) )
def _check3 ( res ) :
2008-11-07 05:35:47 +00:00
self . failUnless ( " Healthy : " in res )
2009-04-07 18:54:59 +00:00
self . failUnless ( " Return to file " in res )
2008-07-16 22:42:56 +00:00
self . failUnless ( redir_url in res )
d . addCallback ( _check3 )
2008-10-24 00:11:18 +00:00
d . addCallback ( lambda res :
self . POST ( bar_url , t = " check " , output = " JSON " ) )
def _check_json ( res ) :
data = simplejson . loads ( res )
self . failUnless ( " storage-index " in data )
self . failUnless ( data [ " results " ] [ " healthy " ] )
d . addCallback ( _check_json )
2008-07-16 22:42:56 +00:00
return d
2008-09-07 19:44:56 +00:00
def test_POST_FILEURL_check_and_repair ( self ) :
bar_url = self . public_url + " /foo/bar.txt "
d = self . POST ( bar_url , t = " check " , repair = " true " )
def _check ( res ) :
2008-11-07 05:35:47 +00:00
self . failUnless ( " Healthy : " in res )
2008-09-07 19:44:56 +00:00
d . addCallback ( _check )
redir_url = " http://allmydata.org/TARGET "
def _check2 ( statuscode , target ) :
self . failUnlessEqual ( statuscode , str ( http . FOUND ) )
self . failUnlessEqual ( target , redir_url )
d . addCallback ( lambda res :
self . shouldRedirect2 ( " test_POST_FILEURL_check_and_repair " ,
_check2 ,
self . POST , bar_url ,
t = " check " , repair = " true " ,
when_done = redir_url ) )
d . addCallback ( lambda res :
self . POST ( bar_url , t = " check " , return_to = redir_url ) )
def _check3 ( res ) :
2008-11-07 05:35:47 +00:00
self . failUnless ( " Healthy : " in res )
2009-04-07 18:54:59 +00:00
self . failUnless ( " Return to file " in res )
2008-09-07 19:44:56 +00:00
self . failUnless ( redir_url in res )
d . addCallback ( _check3 )
return d
2008-07-16 22:42:56 +00:00
def test_POST_DIRURL_check ( self ) :
foo_url = self . public_url + " /foo/ "
d = self . POST ( foo_url , t = " check " )
def _check ( res ) :
2008-11-07 05:35:47 +00:00
self . failUnless ( " Healthy : " in res , res )
2008-07-16 22:42:56 +00:00
d . addCallback ( _check )
redir_url = " http://allmydata.org/TARGET "
def _check2 ( statuscode , target ) :
self . failUnlessEqual ( statuscode , str ( http . FOUND ) )
self . failUnlessEqual ( target , redir_url )
d . addCallback ( lambda res :
self . shouldRedirect2 ( " test_POST_DIRURL_check " ,
_check2 ,
self . POST , foo_url ,
t = " check " ,
when_done = redir_url ) )
d . addCallback ( lambda res :
self . POST ( foo_url , t = " check " , return_to = redir_url ) )
def _check3 ( res ) :
2008-11-07 05:35:47 +00:00
self . failUnless ( " Healthy : " in res , res )
2009-04-07 18:54:59 +00:00
self . failUnless ( " Return to file/directory " in res )
2008-07-16 22:42:56 +00:00
self . failUnless ( redir_url in res )
d . addCallback ( _check3 )
2008-10-24 00:11:18 +00:00
d . addCallback ( lambda res :
self . POST ( foo_url , t = " check " , output = " JSON " ) )
def _check_json ( res ) :
data = simplejson . loads ( res )
self . failUnless ( " storage-index " in data )
self . failUnless ( data [ " results " ] [ " healthy " ] )
d . addCallback ( _check_json )
2008-05-20 18:33:14 +00:00
return d
2008-09-07 19:44:56 +00:00
def test_POST_DIRURL_check_and_repair ( self ) :
foo_url = self . public_url + " /foo/ "
d = self . POST ( foo_url , t = " check " , repair = " true " )
def _check ( res ) :
2008-11-07 05:35:47 +00:00
self . failUnless ( " Healthy : " in res , res )
2008-09-07 19:44:56 +00:00
d . addCallback ( _check )
redir_url = " http://allmydata.org/TARGET "
def _check2 ( statuscode , target ) :
self . failUnlessEqual ( statuscode , str ( http . FOUND ) )
self . failUnlessEqual ( target , redir_url )
d . addCallback ( lambda res :
self . shouldRedirect2 ( " test_POST_DIRURL_check_and_repair " ,
_check2 ,
self . POST , foo_url ,
t = " check " , repair = " true " ,
when_done = redir_url ) )
d . addCallback ( lambda res :
self . POST ( foo_url , t = " check " , return_to = redir_url ) )
def _check3 ( res ) :
2008-11-07 05:35:47 +00:00
self . failUnless ( " Healthy : " in res )
2009-04-07 18:54:59 +00:00
self . failUnless ( " Return to file/directory " in res )
2008-09-07 19:44:56 +00:00
self . failUnless ( redir_url in res )
d . addCallback ( _check3 )
return d
2008-10-22 00:03:07 +00:00
def wait_for_operation ( self , ignored , ophandle ) :
url = " /operations/ " + ophandle
url + = " ?t=status&output=JSON "
d = self . GET ( url )
def _got ( res ) :
data = simplejson . loads ( res )
if not data [ " finished " ] :
d = self . stall ( delay = 1.0 )
d . addCallback ( self . wait_for_operation , ophandle )
return d
return data
d . addCallback ( _got )
return d
def get_operation_results ( self , ignored , ophandle , output = None ) :
url = " /operations/ " + ophandle
url + = " ?t=status "
if output :
url + = " &output= " + output
d = self . GET ( url )
def _got ( res ) :
if output and output . lower ( ) == " json " :
return simplejson . loads ( res )
return res
d . addCallback ( _got )
return d
def test_POST_DIRURL_deepcheck_no_ophandle ( self ) :
d = self . shouldFail2 ( error . Error ,
" test_POST_DIRURL_deepcheck_no_ophandle " ,
" 400 Bad Request " ,
" slow operation requires ophandle= " ,
self . POST , self . public_url , t = " start-deep-check " )
return d
2008-07-17 23:47:09 +00:00
def test_POST_DIRURL_deepcheck ( self ) :
2008-10-22 00:03:07 +00:00
def _check_redirect ( statuscode , target ) :
self . failUnlessEqual ( statuscode , str ( http . FOUND ) )
2008-10-23 22:56:58 +00:00
self . failUnless ( target . endswith ( " /operations/123 " ) )
2008-10-22 00:03:07 +00:00
d = self . shouldRedirect2 ( " test_POST_DIRURL_deepcheck " , _check_redirect ,
self . POST , self . public_url ,
t = " start-deep-check " , ophandle = " 123 " )
d . addCallback ( self . wait_for_operation , " 123 " )
def _check_json ( data ) :
self . failUnlessEqual ( data [ " finished " ] , True )
self . failUnlessEqual ( data [ " count-objects-checked " ] , 8 )
self . failUnlessEqual ( data [ " count-objects-healthy " ] , 8 )
d . addCallback ( _check_json )
d . addCallback ( self . get_operation_results , " 123 " , " html " )
def _check_html ( res ) :
2008-07-17 23:47:09 +00:00
self . failUnless ( " Objects Checked: <span>8</span> " in res )
self . failUnless ( " Objects Healthy: <span>8</span> " in res )
2008-10-22 00:03:07 +00:00
d . addCallback ( _check_html )
2008-10-24 02:17:59 +00:00
d . addCallback ( lambda res :
self . GET ( " /operations/123/ " ) )
d . addCallback ( _check_html ) # should be the same as without the slash
2008-10-24 00:11:18 +00:00
d . addCallback ( lambda res :
self . shouldFail2 ( error . Error , " one " , " 404 Not Found " ,
" No detailed results for SI bogus " ,
self . GET , " /operations/123/bogus " ) )
2008-10-24 02:17:59 +00:00
2008-10-24 00:11:18 +00:00
foo_si = self . _foo_node . get_storage_index ( )
foo_si_s = base32 . b2a ( foo_si )
d . addCallback ( lambda res :
self . GET ( " /operations/123/ %s ?output=JSON " % foo_si_s ) )
def _check_foo_json ( res ) :
data = simplejson . loads ( res )
self . failUnlessEqual ( data [ " storage-index " ] , foo_si_s )
self . failUnless ( data [ " results " ] [ " healthy " ] )
d . addCallback ( _check_foo_json )
2008-09-07 19:44:56 +00:00
return d
def test_POST_DIRURL_deepcheck_and_repair ( self ) :
2008-10-22 00:03:07 +00:00
d = self . POST ( self . public_url , t = " start-deep-check " , repair = " true " ,
ophandle = " 124 " , output = " json " , followRedirect = True )
d . addCallback ( self . wait_for_operation , " 124 " )
def _check_json ( data ) :
self . failUnlessEqual ( data [ " finished " ] , True )
self . failUnlessEqual ( data [ " count-objects-checked " ] , 8 )
self . failUnlessEqual ( data [ " count-objects-healthy-pre-repair " ] , 8 )
self . failUnlessEqual ( data [ " count-objects-unhealthy-pre-repair " ] , 0 )
self . failUnlessEqual ( data [ " count-corrupt-shares-pre-repair " ] , 0 )
self . failUnlessEqual ( data [ " count-repairs-attempted " ] , 0 )
self . failUnlessEqual ( data [ " count-repairs-successful " ] , 0 )
self . failUnlessEqual ( data [ " count-repairs-unsuccessful " ] , 0 )
self . failUnlessEqual ( data [ " count-objects-healthy-post-repair " ] , 8 )
self . failUnlessEqual ( data [ " count-objects-unhealthy-post-repair " ] , 0 )
self . failUnlessEqual ( data [ " count-corrupt-shares-post-repair " ] , 0 )
d . addCallback ( _check_json )
d . addCallback ( self . get_operation_results , " 124 " , " html " )
def _check_html ( res ) :
2008-09-07 19:44:56 +00:00
self . failUnless ( " Objects Checked: <span>8</span> " in res )
self . failUnless ( " Objects Healthy (before repair): <span>8</span> " in res )
self . failUnless ( " Objects Unhealthy (before repair): <span>0</span> " in res )
self . failUnless ( " Corrupt Shares (before repair): <span>0</span> " in res )
2008-07-17 23:47:09 +00:00
self . failUnless ( " Repairs Attempted: <span>0</span> " in res )
self . failUnless ( " Repairs Successful: <span>0</span> " in res )
2008-09-07 19:44:56 +00:00
self . failUnless ( " Repairs Unsuccessful: <span>0</span> " in res )
self . failUnless ( " Objects Healthy (after repair): <span>8</span> " in res )
self . failUnless ( " Objects Unhealthy (after repair): <span>0</span> " in res )
self . failUnless ( " Corrupt Shares (after repair): <span>0</span> " in res )
2008-10-22 00:03:07 +00:00
d . addCallback ( _check_html )
2008-07-17 23:47:09 +00:00
return d
2008-05-20 18:33:14 +00:00
def test_POST_FILEURL_bad_t ( self ) :
d = self . shouldFail2 ( error . Error , " POST_bad_t " , " 400 Bad Request " ,
" POST to file: bad t=bogus " ,
self . POST , self . public_url + " /foo/bar.txt " ,
t = " bogus " )
return d
2007-07-25 10:21:30 +00:00
def test_POST_mkdir ( self ) : # return value?
2007-12-03 21:52:42 +00:00
d = self . POST ( self . public_url + " /foo " , t = " mkdir " , name = " newdir " )
2008-02-14 22:45:56 +00:00
d . addCallback ( lambda res : self . _foo_node . get ( u " newdir " ) )
2007-12-05 06:01:37 +00:00
d . addCallback ( self . failUnlessNodeKeysAre , [ ] )
2007-07-07 02:43:55 +00:00
return d
2008-05-19 19:56:02 +00:00
def test_POST_mkdir_2 ( self ) :
d = self . POST ( self . public_url + " /foo/newdir?t=mkdir " , " " )
d . addCallback ( lambda res :
self . failUnlessNodeHasChild ( self . _foo_node , u " newdir " ) )
d . addCallback ( lambda res : self . _foo_node . get ( u " newdir " ) )
d . addCallback ( self . failUnlessNodeKeysAre , [ ] )
return d
def test_POST_mkdirs_2 ( self ) :
d = self . POST ( self . public_url + " /foo/bardir/newdir?t=mkdir " , " " )
d . addCallback ( lambda res :
self . failUnlessNodeHasChild ( self . _foo_node , u " bardir " ) )
d . addCallback ( lambda res : self . _foo_node . get ( u " bardir " ) )
d . addCallback ( lambda bardirnode : bardirnode . get ( u " newdir " ) )
d . addCallback ( self . failUnlessNodeKeysAre , [ ] )
return d
2007-12-19 00:47:49 +00:00
def test_POST_mkdir_no_parentdir_noredirect ( self ) :
2007-12-25 00:46:52 +00:00
d = self . POST ( " /uri?t=mkdir " )
2007-12-18 20:15:08 +00:00
def _after_mkdir ( res ) :
2008-01-03 23:55:43 +00:00
uri . NewDirectoryURI . init_from_string ( res )
2007-12-18 20:15:08 +00:00
d . addCallback ( _after_mkdir )
return d
2007-12-20 19:58:17 +00:00
def test_POST_mkdir_no_parentdir_redirect ( self ) :
2007-12-25 00:46:52 +00:00
d = self . POST ( " /uri?t=mkdir&redirect_to_result=true " )
2007-12-20 19:58:17 +00:00
d . addBoth ( self . shouldRedirect , None , statuscode = ' 303 ' )
2007-12-25 00:46:52 +00:00
def _check_target ( target ) :
target = urllib . unquote ( target )
self . failUnless ( target . startswith ( " uri/URI:DIR2: " ) , target )
d . addCallback ( _check_target )
2007-12-20 19:58:17 +00:00
return d
2007-12-25 05:49:35 +00:00
def test_POST_noparent_bad ( self ) :
2009-03-03 23:46:04 +00:00
d = self . shouldHTTPError ( " POST /uri?t=bogus " , 400 , " Bad Request " ,
" /uri accepts only PUT, PUT?t=mkdir, "
" POST?t=upload, and POST?t=mkdir " ,
self . POST , " /uri?t=bogus " )
2007-12-25 05:49:35 +00:00
return d
2007-12-20 22:31:12 +00:00
def test_welcome_page_mkdir_button ( self ) :
# Fetch the welcome page.
d = self . GET ( " / " )
def _after_get_welcome_page ( res ) :
2009-05-26 23:25:45 +00:00
MKDIR_BUTTON_RE = re . compile ( ' <form action= " ([^ " ]*) " method= " post " .*?<input type= " hidden " name= " t " value= " ([^ " ]*) " /><input type= " hidden " name= " ([^ " ]*) " value= " ([^ " ]*) " /><input type= " submit " value= " Create a directory " /> ' , re . I )
2007-12-20 22:31:12 +00:00
mo = MKDIR_BUTTON_RE . search ( res )
formaction = mo . group ( 1 )
formt = mo . group ( 2 )
formaname = mo . group ( 3 )
formavalue = mo . group ( 4 )
return ( formaction , formt , formaname , formavalue )
d . addCallback ( _after_get_welcome_page )
def _after_parse_form ( res ) :
( formaction , formt , formaname , formavalue ) = res
return self . POST ( " / %s ?t= %s & %s = %s " % ( formaction , formt , formaname , formavalue ) )
d . addCallback ( _after_parse_form )
d . addBoth ( self . shouldRedirect , None , statuscode = ' 303 ' )
return d
2007-08-15 22:21:38 +00:00
def test_POST_mkdir_replace ( self ) : # return value?
2007-12-03 21:52:42 +00:00
d = self . POST ( self . public_url + " /foo " , t = " mkdir " , name = " sub " )
2008-02-14 22:45:56 +00:00
d . addCallback ( lambda res : self . _foo_node . get ( u " sub " ) )
2007-12-05 06:01:37 +00:00
d . addCallback ( self . failUnlessNodeKeysAre , [ ] )
2007-08-15 22:21:38 +00:00
return d
def test_POST_mkdir_no_replace_queryarg ( self ) : # return value?
2007-12-03 21:52:42 +00:00
d = self . POST ( self . public_url + " /foo?replace=false " , t = " mkdir " , name = " sub " )
2007-08-15 22:21:38 +00:00
d . addBoth ( self . shouldFail , error . Error ,
" POST_mkdir_no_replace_queryarg " ,
" 409 Conflict " ,
" There was already a child by that name, and you asked me "
" to not replace it " )
2008-02-14 22:45:56 +00:00
d . addCallback ( lambda res : self . _foo_node . get ( u " sub " ) )
d . addCallback ( self . failUnlessNodeKeysAre , [ u " baz.txt " ] )
2007-08-15 22:21:38 +00:00
return d
def test_POST_mkdir_no_replace_field ( self ) : # return value?
2007-12-03 21:52:42 +00:00
d = self . POST ( self . public_url + " /foo " , t = " mkdir " , name = " sub " ,
2007-08-15 22:21:38 +00:00
replace = " false " )
d . addBoth ( self . shouldFail , error . Error , " POST_mkdir_no_replace_field " ,
" 409 Conflict " ,
" There was already a child by that name, and you asked me "
" to not replace it " )
2008-02-14 22:45:56 +00:00
d . addCallback ( lambda res : self . _foo_node . get ( u " sub " ) )
d . addCallback ( self . failUnlessNodeKeysAre , [ u " baz.txt " ] )
2007-08-15 22:21:38 +00:00
return d
2007-08-14 00:45:02 +00:00
def test_POST_mkdir_whendone_field ( self ) :
2007-12-03 21:52:42 +00:00
d = self . POST ( self . public_url + " /foo " ,
2007-08-14 00:45:02 +00:00
t = " mkdir " , name = " newdir " , when_done = " /THERE " )
d . addBoth ( self . shouldRedirect , " /THERE " )
2008-02-14 22:45:56 +00:00
d . addCallback ( lambda res : self . _foo_node . get ( u " newdir " ) )
2007-12-05 06:01:37 +00:00
d . addCallback ( self . failUnlessNodeKeysAre , [ ] )
2007-08-14 00:45:02 +00:00
return d
def test_POST_mkdir_whendone_queryarg ( self ) :
2007-12-03 21:52:42 +00:00
d = self . POST ( self . public_url + " /foo?when_done=/THERE " ,
2007-07-08 05:47:18 +00:00
t = " mkdir " , name = " newdir " )
d . addBoth ( self . shouldRedirect , " /THERE " )
2008-02-14 22:45:56 +00:00
d . addCallback ( lambda res : self . _foo_node . get ( u " newdir " ) )
2007-12-05 06:01:37 +00:00
d . addCallback ( self . failUnlessNodeKeysAre , [ ] )
2007-07-08 05:47:18 +00:00
return d
2008-04-15 18:11:29 +00:00
def test_POST_bad_t ( self ) :
d = self . shouldFail2 ( error . Error , " POST_bad_t " , " 400 Bad Request " ,
2008-05-19 19:56:02 +00:00
" POST to a directory with bad t=BOGUS " ,
2008-04-15 18:11:29 +00:00
self . POST , self . public_url + " /foo " , t = " BOGUS " )
return d
2008-03-01 01:40:27 +00:00
def test_POST_set_children ( self ) :
contents9 , n9 , newuri9 = self . makefile ( 9 )
contents10 , n10 , newuri10 = self . makefile ( 10 )
contents11 , n11 , newuri11 = self . makefile ( 11 )
reqbody = """ {
" atomic_added_1 " : [ " filenode " , { " rw_uri " : " %s " ,
" size " : 0 ,
" metadata " : {
" ctime " : 1002777696.7564139 ,
" mtime " : 1002777696.7564139
}
2008-03-03 21:48:52 +00:00
} ] ,
2008-03-01 01:40:27 +00:00
" atomic_added_2 " : [ " filenode " , { " rw_uri " : " %s " ,
" size " : 1 ,
" metadata " : {
" ctime " : 1002777696.7564139 ,
" mtime " : 1002777696.7564139
}
} ] ,
" atomic_added_3 " : [ " filenode " , { " rw_uri " : " %s " ,
" size " : 2 ,
" metadata " : {
" ctime " : 1002777696.7564139 ,
" mtime " : 1002777696.7564139
}
} ]
} """ % (newuri9, newuri10, newuri11)
url = self . webish_url + self . public_url + " /foo " + " ?t=set_children "
d = client . getPage ( url , method = " POST " , postdata = reqbody )
def _then ( res ) :
self . failUnlessURIMatchesChild ( newuri9 , self . _foo_node , u " atomic_added_1 " )
self . failUnlessURIMatchesChild ( newuri10 , self . _foo_node , u " atomic_added_2 " )
self . failUnlessURIMatchesChild ( newuri11 , self . _foo_node , u " atomic_added_3 " )
d . addCallback ( _then )
2008-05-19 19:33:39 +00:00
d . addErrback ( self . dump_error )
2008-03-01 01:40:27 +00:00
return d
2007-07-25 03:16:21 +00:00
def test_POST_put_uri ( self ) :
2007-12-05 06:01:37 +00:00
contents , n , newuri = self . makefile ( 8 )
2007-12-03 21:52:42 +00:00
d = self . POST ( self . public_url + " /foo " , t = " uri " , name = " new.txt " , uri = newuri )
2008-02-14 22:45:56 +00:00
d . addCallback ( self . failUnlessURIMatchesChild , self . _foo_node , u " new.txt " )
2007-12-05 06:01:37 +00:00
d . addCallback ( lambda res :
2008-02-14 22:45:56 +00:00
self . failUnlessChildContentsAre ( self . _foo_node , u " new.txt " ,
2007-12-05 06:01:37 +00:00
contents ) )
2007-07-07 02:43:55 +00:00
return d
2007-08-15 22:21:38 +00:00
def test_POST_put_uri_replace ( self ) :
2007-12-05 06:01:37 +00:00
contents , n , newuri = self . makefile ( 8 )
2007-12-03 21:52:42 +00:00
d = self . POST ( self . public_url + " /foo " , t = " uri " , name = " bar.txt " , uri = newuri )
2008-02-14 22:45:56 +00:00
d . addCallback ( self . failUnlessURIMatchesChild , self . _foo_node , u " bar.txt " )
2007-12-05 06:01:37 +00:00
d . addCallback ( lambda res :
2008-02-14 22:45:56 +00:00
self . failUnlessChildContentsAre ( self . _foo_node , u " bar.txt " ,
2007-12-05 06:01:37 +00:00
contents ) )
2007-08-15 22:21:38 +00:00
return d
def test_POST_put_uri_no_replace_queryarg ( self ) :
2007-12-05 06:01:37 +00:00
contents , n , newuri = self . makefile ( 8 )
2007-12-03 21:52:42 +00:00
d = self . POST ( self . public_url + " /foo?replace=false " , t = " uri " ,
2007-08-15 22:21:38 +00:00
name = " bar.txt " , uri = newuri )
d . addBoth ( self . shouldFail , error . Error ,
" POST_put_uri_no_replace_queryarg " ,
" 409 Conflict " ,
" There was already a child by that name, and you asked me "
" to not replace it " )
2007-12-03 21:52:42 +00:00
d . addCallback ( lambda res : self . GET ( self . public_url + " /foo/bar.txt " ) )
2007-08-15 22:21:38 +00:00
d . addCallback ( self . failUnlessIsBarDotTxt )
return d
def test_POST_put_uri_no_replace_field ( self ) :
2007-12-05 06:01:37 +00:00
contents , n , newuri = self . makefile ( 8 )
2007-12-03 21:52:42 +00:00
d = self . POST ( self . public_url + " /foo " , t = " uri " , replace = " false " ,
2007-08-15 22:21:38 +00:00
name = " bar.txt " , uri = newuri )
d . addBoth ( self . shouldFail , error . Error ,
" POST_put_uri_no_replace_field " ,
" 409 Conflict " ,
" There was already a child by that name, and you asked me "
" to not replace it " )
2007-12-03 21:52:42 +00:00
d . addCallback ( lambda res : self . GET ( self . public_url + " /foo/bar.txt " ) )
2007-08-15 22:21:38 +00:00
d . addCallback ( self . failUnlessIsBarDotTxt )
return d
2007-07-25 03:16:21 +00:00
def test_POST_delete ( self ) :
2007-12-03 21:52:42 +00:00
d = self . POST ( self . public_url + " /foo " , t = " delete " , name = " bar.txt " )
2007-12-05 06:01:37 +00:00
d . addCallback ( lambda res : self . _foo_node . list ( ) )
def _check ( children ) :
2008-02-14 22:45:56 +00:00
self . failIf ( u " bar.txt " in children )
2007-07-08 03:06:58 +00:00
d . addCallback ( _check )
2007-07-07 02:43:55 +00:00
return d
2007-07-25 03:16:21 +00:00
def test_POST_rename_file ( self ) :
2007-12-03 21:52:42 +00:00
d = self . POST ( self . public_url + " /foo " , t = " rename " ,
2007-07-17 00:05:01 +00:00
from_name = " bar.txt " , to_name = ' wibble.txt ' )
2007-12-05 06:01:37 +00:00
d . addCallback ( lambda res :
2008-02-14 22:45:56 +00:00
self . failIfNodeHasChild ( self . _foo_node , u " bar.txt " ) )
2007-12-05 06:01:37 +00:00
d . addCallback ( lambda res :
2008-02-14 22:45:56 +00:00
self . failUnlessNodeHasChild ( self . _foo_node , u " wibble.txt " ) )
2007-12-03 21:52:42 +00:00
d . addCallback ( lambda res : self . GET ( self . public_url + " /foo/wibble.txt " ) )
2007-07-17 00:05:01 +00:00
d . addCallback ( self . failUnlessIsBarDotTxt )
2007-12-03 21:52:42 +00:00
d . addCallback ( lambda res : self . GET ( self . public_url + " /foo/wibble.txt?t=json " ) )
2007-07-17 00:05:01 +00:00
d . addCallback ( self . failUnlessIsBarJSON )
return d
2008-10-23 23:32:36 +00:00
def test_POST_rename_file_redundant ( self ) :
d = self . POST ( self . public_url + " /foo " , t = " rename " ,
from_name = " bar.txt " , to_name = ' bar.txt ' )
d . addCallback ( lambda res :
self . failUnlessNodeHasChild ( self . _foo_node , u " bar.txt " ) )
d . addCallback ( lambda res : self . GET ( self . public_url + " /foo/bar.txt " ) )
d . addCallback ( self . failUnlessIsBarDotTxt )
d . addCallback ( lambda res : self . GET ( self . public_url + " /foo/bar.txt?t=json " ) )
d . addCallback ( self . failUnlessIsBarJSON )
return d
2007-08-15 22:21:38 +00:00
def test_POST_rename_file_replace ( self ) :
# rename a file and replace a directory with it
2007-12-03 21:52:42 +00:00
d = self . POST ( self . public_url + " /foo " , t = " rename " ,
2007-08-15 22:21:38 +00:00
from_name = " bar.txt " , to_name = ' empty ' )
2007-12-05 06:01:37 +00:00
d . addCallback ( lambda res :
2008-02-14 22:45:56 +00:00
self . failIfNodeHasChild ( self . _foo_node , u " bar.txt " ) )
2007-12-05 06:01:37 +00:00
d . addCallback ( lambda res :
2008-02-14 22:45:56 +00:00
self . failUnlessNodeHasChild ( self . _foo_node , u " empty " ) )
2007-12-03 21:52:42 +00:00
d . addCallback ( lambda res : self . GET ( self . public_url + " /foo/empty " ) )
2007-08-15 22:21:38 +00:00
d . addCallback ( self . failUnlessIsBarDotTxt )
2007-12-03 21:52:42 +00:00
d . addCallback ( lambda res : self . GET ( self . public_url + " /foo/empty?t=json " ) )
2007-08-15 22:21:38 +00:00
d . addCallback ( self . failUnlessIsBarJSON )
return d
def test_POST_rename_file_no_replace_queryarg ( self ) :
# rename a file and replace a directory with it
2007-12-03 21:52:42 +00:00
d = self . POST ( self . public_url + " /foo?replace=false " , t = " rename " ,
2007-08-15 22:21:38 +00:00
from_name = " bar.txt " , to_name = ' empty ' )
d . addBoth ( self . shouldFail , error . Error ,
" POST_rename_file_no_replace_queryarg " ,
" 409 Conflict " ,
" There was already a child by that name, and you asked me "
" to not replace it " )
2007-12-03 21:52:42 +00:00
d . addCallback ( lambda res : self . GET ( self . public_url + " /foo/empty?t=json " ) )
2007-08-15 22:21:38 +00:00
d . addCallback ( self . failUnlessIsEmptyJSON )
return d
def test_POST_rename_file_no_replace_field ( self ) :
# rename a file and replace a directory with it
2007-12-03 21:52:42 +00:00
d = self . POST ( self . public_url + " /foo " , t = " rename " , replace = " false " ,
2007-08-15 22:21:38 +00:00
from_name = " bar.txt " , to_name = ' empty ' )
d . addBoth ( self . shouldFail , error . Error ,
" POST_rename_file_no_replace_field " ,
" 409 Conflict " ,
" There was already a child by that name, and you asked me "
" to not replace it " )
2007-12-03 21:52:42 +00:00
d . addCallback ( lambda res : self . GET ( self . public_url + " /foo/empty?t=json " ) )
2007-08-15 22:21:38 +00:00
d . addCallback ( self . failUnlessIsEmptyJSON )
return d
def failUnlessIsEmptyJSON ( self , res ) :
2007-12-05 06:42:54 +00:00
data = simplejson . loads ( res )
2007-12-03 21:52:42 +00:00
self . failUnlessEqual ( data [ 0 ] , " dirnode " , data )
2007-08-15 22:21:38 +00:00
self . failUnlessEqual ( len ( data [ 1 ] [ " children " ] ) , 0 )
2007-07-25 03:16:21 +00:00
def test_POST_rename_file_slash_fail ( self ) :
2007-12-03 21:52:42 +00:00
d = self . POST ( self . public_url + " /foo " , t = " rename " ,
2007-07-17 00:05:01 +00:00
from_name = " bar.txt " , to_name = ' kirk/spock.txt ' )
d . addBoth ( self . shouldFail , error . Error ,
" test_POST_rename_file_slash_fail " ,
" 400 Bad Request " ,
" to_name= may not contain a slash " ,
)
2007-12-05 06:01:37 +00:00
d . addCallback ( lambda res :
2008-02-14 22:45:56 +00:00
self . failUnlessNodeHasChild ( self . _foo_node , u " bar.txt " ) )
2007-07-17 00:05:01 +00:00
return d
2007-07-25 03:16:21 +00:00
def test_POST_rename_dir ( self ) :
2007-12-03 21:52:42 +00:00
d = self . POST ( self . public_url , t = " rename " ,
2007-07-17 00:05:01 +00:00
from_name = " foo " , to_name = ' plunk ' )
2007-12-05 06:01:37 +00:00
d . addCallback ( lambda res :
2008-02-14 22:45:56 +00:00
self . failIfNodeHasChild ( self . public_root , u " foo " ) )
2007-12-05 06:01:37 +00:00
d . addCallback ( lambda res :
2008-02-14 22:45:56 +00:00
self . failUnlessNodeHasChild ( self . public_root , u " plunk " ) )
2007-12-03 21:52:42 +00:00
d . addCallback ( lambda res : self . GET ( self . public_url + " /plunk?t=json " ) )
2007-07-17 00:05:01 +00:00
d . addCallback ( self . failUnlessIsFooJSON )
return d
2008-06-18 02:49:40 +00:00
def shouldRedirect ( self , res , target = None , statuscode = None , which = " " ) :
2007-12-20 19:58:17 +00:00
""" If target is not None then the redirection has to go to target. If
statuscode is not None then the redirection has to be accomplished with
that HTTP status code . """
2007-07-08 03:06:58 +00:00
if not isinstance ( res , failure . Failure ) :
2008-06-18 02:49:40 +00:00
to_where = ( target is None ) and " somewhere " or ( " to " + target )
self . fail ( " %s : we were expecting to get redirected %s , not get an "
" actual page: %s " % ( which , to_where , res ) )
2007-07-08 03:06:58 +00:00
res . trap ( error . PageRedirect )
2007-12-20 19:58:17 +00:00
if statuscode is not None :
2008-06-18 02:49:40 +00:00
self . failUnlessEqual ( res . value . status , statuscode ,
" %s : not a redirect " % which )
2007-12-20 19:58:17 +00:00
if target is not None :
# the PageRedirect does not seem to capture the uri= query arg
# properly, so we can't check for it.
realtarget = self . webish_url + target
2008-06-18 02:49:40 +00:00
self . failUnlessEqual ( res . value . location , realtarget ,
" %s : wrong target " % which )
2007-12-25 00:46:52 +00:00
return res . value . location
2007-07-08 03:06:58 +00:00
2007-07-25 03:16:21 +00:00
def test_GET_URI_form ( self ) :
2007-07-08 03:06:58 +00:00
base = " /uri?uri= %s " % self . _bar_txt_uri
# this is supposed to give us a redirect to /uri/$URI, plus arguments
targetbase = " /uri/ %s " % urllib . quote ( self . _bar_txt_uri )
d = self . GET ( base )
d . addBoth ( self . shouldRedirect , targetbase )
d . addCallback ( lambda res : self . GET ( base + " &filename=bar.txt " ) )
d . addBoth ( self . shouldRedirect , targetbase + " ?filename=bar.txt " )
d . addCallback ( lambda res : self . GET ( base + " &t=json " ) )
d . addBoth ( self . shouldRedirect , targetbase + " ?t=json " )
d . addCallback ( self . log , " about to get file by uri " )
d . addCallback ( lambda res : self . GET ( base , followRedirect = True ) )
d . addCallback ( self . failUnlessIsBarDotTxt )
d . addCallback ( self . log , " got file by uri, about to get dir by uri " )
d . addCallback ( lambda res : self . GET ( " /uri?uri= %s &t=json " % self . _foo_uri ,
followRedirect = True ) )
d . addCallback ( self . failUnlessIsFooJSON )
d . addCallback ( self . log , " got dir by uri " )
return d
2008-05-20 06:28:52 +00:00
def test_GET_URI_form_bad ( self ) :
d = self . shouldFail2 ( error . Error , " test_GET_URI_form_bad " ,
" 400 Bad Request " , " GET /uri requires uri= " ,
self . GET , " /uri " )
return d
2007-07-25 03:16:21 +00:00
def test_GET_rename_form ( self ) :
2007-12-03 21:52:42 +00:00
d = self . GET ( self . public_url + " /foo?t=rename-form&name=bar.txt " ,
2008-06-18 02:49:40 +00:00
followRedirect = True )
2007-07-17 00:05:01 +00:00
def _check ( res ) :
2008-06-18 02:49:40 +00:00
self . failUnless ( ' name= " when_done " value= " . " ' in res , res )
2007-07-17 00:05:01 +00:00
self . failUnless ( re . search ( r ' name= " from_name " value= " bar \ .txt " ' , res ) )
d . addCallback ( _check )
return d
2007-07-08 03:06:58 +00:00
def log ( self , res , msg ) :
2007-07-08 03:11:30 +00:00
#print "MSG: %s RES: %s" % (msg, res)
2007-07-08 03:06:58 +00:00
log . msg ( msg )
return res
2007-07-25 03:16:21 +00:00
def test_GET_URI_URL ( self ) :
2007-12-20 00:54:40 +00:00
base = " /uri/ %s " % self . _bar_txt_uri
2007-07-08 03:06:58 +00:00
d = self . GET ( base )
d . addCallback ( self . failUnlessIsBarDotTxt )
d . addCallback ( lambda res : self . GET ( base + " ?filename=bar.txt " ) )
d . addCallback ( self . failUnlessIsBarDotTxt )
d . addCallback ( lambda res : self . GET ( base + " ?filename=bar.txt&save=true " ) )
d . addCallback ( self . failUnlessIsBarDotTxt )
return d
2007-07-25 03:16:21 +00:00
def test_GET_URI_URL_dir ( self ) :
2007-12-20 00:54:40 +00:00
base = " /uri/ %s ?t=json " % self . _foo_uri
2007-07-08 03:06:58 +00:00
d = self . GET ( base )
d . addCallback ( self . failUnlessIsFooJSON )
return d
2007-07-25 03:16:21 +00:00
def test_GET_URI_URL_missing ( self ) :
2007-12-20 00:54:40 +00:00
base = " /uri/ %s " % self . _bad_file_uri
2009-03-03 23:46:04 +00:00
d = self . shouldHTTPError ( " test_GET_URI_URL_missing " ,
http . GONE , None , " NotEnoughSharesError " ,
self . GET , base )
2007-07-17 19:16:45 +00:00
# TODO: how can we exercise both sides of WebDownloadTarget.fail
# here? we must arrange for a download to fail after target.open()
# has been called, and then inspect the response to see that it is
# shorter than we expected.
2007-07-08 05:06:52 +00:00
return d
2008-10-29 04:54:46 +00:00
def test_PUT_DIRURL_uri ( self ) :
d = self . s . create_empty_dirnode ( )
def _made_dir ( dn ) :
new_uri = dn . get_uri ( )
# replace /foo with a new (empty) directory
d = self . PUT ( self . public_url + " /foo?t=uri " , new_uri )
d . addCallback ( lambda res :
self . failUnlessEqual ( res . strip ( ) , new_uri ) )
d . addCallback ( lambda res :
self . failUnlessChildURIIs ( self . public_root ,
u " foo " ,
new_uri ) )
return d
d . addCallback ( _made_dir )
return d
2008-10-29 04:57:44 +00:00
def test_PUT_DIRURL_uri_noreplace ( self ) :
d = self . s . create_empty_dirnode ( )
def _made_dir ( dn ) :
new_uri = dn . get_uri ( )
# replace /foo with a new (empty) directory, but ask that
# replace=false, so it should fail
d = self . shouldFail2 ( error . Error , " test_PUT_DIRURL_uri_noreplace " ,
" 409 Conflict " , " There was already a child by that name, and you asked me to not replace it " ,
self . PUT ,
self . public_url + " /foo?t=uri&replace=false " ,
new_uri )
d . addCallback ( lambda res :
self . failUnlessChildURIIs ( self . public_root ,
u " foo " ,
self . _foo_uri ) )
return d
d . addCallback ( _made_dir )
return d
2008-10-29 05:00:15 +00:00
def test_PUT_DIRURL_bad_t ( self ) :
d = self . shouldFail2 ( error . Error , " test_PUT_DIRURL_bad_t " ,
" 400 Bad Request " , " PUT to a directory " ,
self . PUT , self . public_url + " /foo?t=BOGUS " , " " )
d . addCallback ( lambda res :
self . failUnlessChildURIIs ( self . public_root ,
u " foo " ,
self . _foo_uri ) )
return d
2007-07-25 03:16:21 +00:00
def test_PUT_NEWFILEURL_uri ( self ) :
2007-12-05 06:01:37 +00:00
contents , n , new_uri = self . makefile ( 8 )
2007-12-03 21:52:42 +00:00
d = self . PUT ( self . public_url + " /foo/new.txt?t=uri " , new_uri )
2007-12-05 06:01:37 +00:00
d . addCallback ( lambda res : self . failUnlessEqual ( res . strip ( ) , new_uri ) )
d . addCallback ( lambda res :
2008-02-14 22:45:56 +00:00
self . failUnlessChildContentsAre ( self . _foo_node , u " new.txt " ,
2007-12-05 06:01:37 +00:00
contents ) )
2007-07-07 02:43:55 +00:00
return d
2007-08-15 20:22:23 +00:00
def test_PUT_NEWFILEURL_uri_replace ( self ) :
2007-12-05 06:01:37 +00:00
contents , n , new_uri = self . makefile ( 8 )
2007-12-03 21:52:42 +00:00
d = self . PUT ( self . public_url + " /foo/bar.txt?t=uri " , new_uri )
2007-12-05 06:01:37 +00:00
d . addCallback ( lambda res : self . failUnlessEqual ( res . strip ( ) , new_uri ) )
d . addCallback ( lambda res :
2008-02-14 22:45:56 +00:00
self . failUnlessChildContentsAre ( self . _foo_node , u " bar.txt " ,
2007-12-05 06:01:37 +00:00
contents ) )
2007-08-15 20:22:23 +00:00
return d
def test_PUT_NEWFILEURL_uri_no_replace ( self ) :
2007-12-05 06:01:37 +00:00
contents , n , new_uri = self . makefile ( 8 )
2007-12-03 21:52:42 +00:00
d = self . PUT ( self . public_url + " /foo/bar.txt?t=uri&replace=false " , new_uri )
2007-08-15 20:22:23 +00:00
d . addBoth ( self . shouldFail , error . Error , " PUT_NEWFILEURL_uri_no_replace " ,
" 409 Conflict " ,
" There was already a child by that name, and you asked me "
" to not replace it " )
return d
2007-09-06 00:12:27 +00:00
def test_PUT_NEWFILE_URI ( self ) :
file_contents = " New file contents here \n "
d = self . PUT ( " /uri " , file_contents )
2009-04-08 02:13:40 +00:00
def _check ( uri ) :
assert isinstance ( uri , str ) , uri
self . failUnless ( uri in FakeCHKFileNode . all_contents )
self . failUnlessEqual ( FakeCHKFileNode . all_contents [ uri ] ,
file_contents )
return self . GET ( " /uri/ %s " % uri )
d . addCallback ( _check )
def _check2 ( res ) :
self . failUnlessEqual ( res , file_contents )
d . addCallback ( _check2 )
return d
def test_PUT_NEWFILE_URI_not_mutable ( self ) :
file_contents = " New file contents here \n "
d = self . PUT ( " /uri?mutable=false " , file_contents )
2007-09-06 00:12:27 +00:00
def _check ( uri ) :
2008-12-19 15:39:24 +00:00
assert isinstance ( uri , str ) , uri
2007-12-05 06:01:37 +00:00
self . failUnless ( uri in FakeCHKFileNode . all_contents )
self . failUnlessEqual ( FakeCHKFileNode . all_contents [ uri ] ,
file_contents )
2007-12-20 00:54:40 +00:00
return self . GET ( " /uri/ %s " % uri )
2007-09-06 00:12:27 +00:00
d . addCallback ( _check )
def _check2 ( res ) :
self . failUnlessEqual ( res , file_contents )
d . addCallback ( _check2 )
return d
2007-09-17 08:53:46 +00:00
def test_PUT_NEWFILE_URI_only_PUT ( self ) :
d = self . PUT ( " /uri?t=bogus " , " " )
d . addBoth ( self . shouldFail , error . Error ,
" PUT_NEWFILE_URI_only_PUT " ,
" 400 Bad Request " ,
2008-05-19 19:56:02 +00:00
" /uri accepts only PUT, PUT?t=mkdir, POST?t=upload, and POST?t=mkdir " )
2007-09-17 08:53:46 +00:00
return d
2008-02-06 05:18:02 +00:00
def test_PUT_NEWFILE_URI_mutable ( self ) :
file_contents = " New file contents here \n "
d = self . PUT ( " /uri?mutable=true " , file_contents )
2008-02-06 05:37:51 +00:00
def _check_mutable ( uri ) :
2008-02-06 05:18:02 +00:00
uri = uri . strip ( )
u = IURI ( uri )
self . failUnless ( IMutableFileURI . providedBy ( u ) )
self . failUnless ( u . storage_index in FakeMutableFileNode . all_contents )
n = self . s . create_node_from_uri ( uri )
2008-04-18 00:51:38 +00:00
return n . download_best_version ( )
2008-02-06 05:37:51 +00:00
d . addCallback ( _check_mutable )
def _check2_mutable ( data ) :
2008-02-06 05:18:02 +00:00
self . failUnlessEqual ( data , file_contents )
2008-02-06 05:37:51 +00:00
d . addCallback ( _check2_mutable )
2008-02-06 05:18:02 +00:00
return d
def _check ( uri ) :
2008-12-19 15:39:24 +00:00
self . failUnless ( uri . to_string ( ) in FakeCHKFileNode . all_contents )
self . failUnlessEqual ( FakeCHKFileNode . all_contents [ uri . to_string ( ) ] ,
2008-02-06 05:18:02 +00:00
file_contents )
return self . GET ( " /uri/ %s " % uri )
d . addCallback ( _check )
def _check2 ( res ) :
self . failUnlessEqual ( res , file_contents )
d . addCallback ( _check2 )
return d
2007-12-20 19:58:17 +00:00
def test_PUT_mkdir ( self ) :
2007-09-06 00:23:06 +00:00
d = self . PUT ( " /uri?t=mkdir " , " " )
def _check ( uri ) :
2007-12-05 06:01:37 +00:00
n = self . s . create_node_from_uri ( uri . strip ( ) )
d2 = self . failUnlessNodeKeysAre ( n , [ ] )
d2 . addCallback ( lambda res :
2007-12-20 00:54:40 +00:00
self . GET ( " /uri/ %s ?t=json " % uri ) )
2007-12-05 06:01:37 +00:00
return d2
2007-09-06 00:23:06 +00:00
d . addCallback ( _check )
d . addCallback ( self . failUnlessIsEmptyJSON )
return d
2007-12-05 06:49:38 +00:00
def test_POST_check ( self ) :
d = self . POST ( self . public_url + " /foo " , t = " check " , name = " bar.txt " )
def _done ( res ) :
# this returns a string form of the results, which are probably
# None since we're using fake filenodes.
# TODO: verify that the check actually happened, by changing
# FakeCHKFileNode to count how many times .check() has been
# called.
pass
d . addCallback ( _done )
return d
2007-12-25 10:48:57 +00:00
def test_bad_method ( self ) :
url = self . webish_url + self . public_url + " /foo/bar.txt "
2009-03-03 23:46:04 +00:00
d = self . shouldHTTPError ( " test_bad_method " ,
501 , " Not Implemented " ,
" I don ' t know how to treat a BOGUS request. " ,
client . getPage , url , method = " BOGUS " )
2007-12-25 10:48:57 +00:00
return d
def test_short_url ( self ) :
url = self . webish_url + " /uri "
2009-03-03 23:46:04 +00:00
d = self . shouldHTTPError ( " test_short_url " , 501 , " Not Implemented " ,
" I don ' t know how to treat a DELETE request. " ,
client . getPage , url , method = " DELETE " )
2007-12-25 10:48:57 +00:00
return d
2008-05-20 22:21:46 +00:00
2008-10-22 05:13:54 +00:00
def test_ophandle_bad ( self ) :
2008-10-22 00:52:56 +00:00
url = self . webish_url + " /operations/bogus?t=status "
2009-03-03 23:46:04 +00:00
d = self . shouldHTTPError ( " test_ophandle_bad " , 404 , " 404 Not Found " ,
" unknown/expired handle ' bogus ' " ,
client . getPage , url )
2008-10-22 00:52:56 +00:00
return d
2008-10-22 05:13:54 +00:00
def test_ophandle_cancel ( self ) :
d = self . POST ( self . public_url + " /foo/?t=start-manifest&ophandle=128 " ,
followRedirect = True )
d . addCallback ( lambda ignored :
self . GET ( " /operations/128?t=status&output=JSON " ) )
def _check1 ( res ) :
data = simplejson . loads ( res )
self . failUnless ( " finished " in data , res )
monitor = self . ws . root . child_operations . handles [ " 128 " ] [ 0 ]
d = self . POST ( " /operations/128?t=cancel&output=JSON " )
def _check2 ( res ) :
data = simplejson . loads ( res )
self . failUnless ( " finished " in data , res )
# t=cancel causes the handle to be forgotten
self . failUnless ( monitor . is_cancelled ( ) )
d . addCallback ( _check2 )
return d
d . addCallback ( _check1 )
d . addCallback ( lambda ignored :
2009-03-03 23:46:04 +00:00
self . shouldHTTPError ( " test_ophandle_cancel " ,
404 , " 404 Not Found " ,
" unknown/expired handle ' 128 ' " ,
self . GET ,
" /operations/128?t=status&output=JSON " ) )
2008-10-22 05:13:54 +00:00
return d
def test_ophandle_retainfor ( self ) :
d = self . POST ( self . public_url + " /foo/?t=start-manifest&ophandle=129&retain-for=60 " ,
followRedirect = True )
d . addCallback ( lambda ignored :
self . GET ( " /operations/129?t=status&output=JSON&retain-for=0 " ) )
def _check1 ( res ) :
data = simplejson . loads ( res )
self . failUnless ( " finished " in data , res )
d . addCallback ( _check1 )
# the retain-for=0 will cause the handle to be expired very soon
d . addCallback ( self . stall , 2.0 )
d . addCallback ( lambda ignored :
2009-03-03 23:46:04 +00:00
self . shouldHTTPError ( " test_ophandle_retainfor " ,
404 , " 404 Not Found " ,
" unknown/expired handle ' 129 ' " ,
self . GET ,
" /operations/129?t=status&output=JSON " ) )
2008-10-22 05:13:54 +00:00
return d
def test_ophandle_release_after_complete ( self ) :
d = self . POST ( self . public_url + " /foo/?t=start-manifest&ophandle=130 " ,
followRedirect = True )
d . addCallback ( self . wait_for_operation , " 130 " )
d . addCallback ( lambda ignored :
self . GET ( " /operations/130?t=status&output=JSON&release-after-complete=true " ) )
# the release-after-complete=true will cause the handle to be expired
d . addCallback ( lambda ignored :
2009-03-03 23:46:04 +00:00
self . shouldHTTPError ( " test_ophandle_release_after_complete " ,
404 , " 404 Not Found " ,
" unknown/expired handle ' 130 ' " ,
self . GET ,
" /operations/130?t=status&output=JSON " ) )
2008-10-22 05:13:54 +00:00
return d
2008-10-22 00:52:56 +00:00
def test_incident ( self ) :
d = self . POST ( " /report_incident " , details = " eek " )
def _done ( res ) :
self . failUnless ( " Thank you for your report! " in res , res )
d . addCallback ( _done )
return d
2008-10-29 22:34:31 +00:00
def test_static ( self ) :
webdir = os . path . join ( self . staticdir , " subdir " )
fileutil . make_dirs ( webdir )
f = open ( os . path . join ( webdir , " hello.txt " ) , " wb " )
f . write ( " hello " )
f . close ( )
d = self . GET ( " /static/subdir/hello.txt " )
def _check ( res ) :
self . failUnlessEqual ( res , " hello " )
d . addCallback ( _check )
return d
2008-10-22 00:52:56 +00:00
2008-05-20 22:21:46 +00:00
class Util ( unittest . TestCase ) :
def test_abbreviate_time ( self ) :
self . failUnlessEqual ( common . abbreviate_time ( None ) , " " )
self . failUnlessEqual ( common . abbreviate_time ( 1.234 ) , " 1.23s " )
self . failUnlessEqual ( common . abbreviate_time ( 0.123 ) , " 123ms " )
self . failUnlessEqual ( common . abbreviate_time ( 0.00123 ) , " 1.2ms " )
self . failUnlessEqual ( common . abbreviate_time ( 0.000123 ) , " 123us " )
def test_abbreviate_rate ( self ) :
self . failUnlessEqual ( common . abbreviate_rate ( None ) , " " )
self . failUnlessEqual ( common . abbreviate_rate ( 1234000 ) , " 1.23MBps " )
self . failUnlessEqual ( common . abbreviate_rate ( 12340 ) , " 12.3kBps " )
self . failUnlessEqual ( common . abbreviate_rate ( 123 ) , " 123Bps " )
def test_abbreviate_size ( self ) :
self . failUnlessEqual ( common . abbreviate_size ( None ) , " " )
self . failUnlessEqual ( common . abbreviate_size ( 1.23 * 1000 * 1000 * 1000 ) , " 1.23GB " )
self . failUnlessEqual ( common . abbreviate_size ( 1.23 * 1000 * 1000 ) , " 1.23MB " )
self . failUnlessEqual ( common . abbreviate_size ( 1230 ) , " 1.2kB " )
self . failUnlessEqual ( common . abbreviate_size ( 123 ) , " 123B " )
2009-02-22 01:01:16 +00:00
def test_plural ( self ) :
def convert ( s ) :
return " %d second %s " % ( s , status . plural ( s ) )
self . failUnlessEqual ( convert ( 0 ) , " 0 seconds " )
self . failUnlessEqual ( convert ( 1 ) , " 1 second " )
self . failUnlessEqual ( convert ( 2 ) , " 2 seconds " )
def convert2 ( s ) :
return " has share %s : %s " % ( status . plural ( s ) , " , " . join ( s ) )
self . failUnlessEqual ( convert2 ( [ ] ) , " has shares: " )
self . failUnlessEqual ( convert2 ( [ " 1 " ] ) , " has share: 1 " )
self . failUnlessEqual ( convert2 ( [ " 1 " , " 2 " ] ) , " has shares: 1,2 " )
2009-02-17 05:12:42 +00:00
2009-02-24 22:40:17 +00:00
class Grid ( GridTestMixin , WebErrorMixin , unittest . TestCase , ShouldFailMixin ) :
2009-02-17 05:12:42 +00:00
2009-02-18 02:32:43 +00:00
def CHECK ( self , ign , which , args , clientnum = 0 ) :
2009-02-17 05:12:42 +00:00
fileurl = self . fileurls [ which ]
2009-02-17 06:00:34 +00:00
url = fileurl + " ? " + args
2009-02-18 02:32:43 +00:00
return self . GET ( url , method = " POST " , clientnum = clientnum )
2009-02-17 05:12:42 +00:00
def test_filecheck ( self ) :
self . basedir = " web/Grid/filecheck "
self . set_up_grid ( )
c0 = self . g . clients [ 0 ]
self . uris = { }
DATA = " data " * 100
d = c0 . upload ( upload . Data ( DATA , convergence = " " ) )
def _stash_uri ( ur , which ) :
self . uris [ which ] = ur . uri
d . addCallback ( _stash_uri , " good " )
d . addCallback ( lambda ign :
c0 . upload ( upload . Data ( DATA + " 1 " , convergence = " " ) ) )
d . addCallback ( _stash_uri , " sick " )
d . addCallback ( lambda ign :
c0 . upload ( upload . Data ( DATA + " 2 " , convergence = " " ) ) )
d . addCallback ( _stash_uri , " dead " )
def _stash_mutable_uri ( n , which ) :
self . uris [ which ] = n . get_uri ( )
assert isinstance ( self . uris [ which ] , str )
d . addCallback ( lambda ign : c0 . create_mutable_file ( DATA + " 3 " ) )
d . addCallback ( _stash_mutable_uri , " corrupt " )
d . addCallback ( lambda ign :
c0 . upload ( upload . Data ( " literal " , convergence = " " ) ) )
d . addCallback ( _stash_uri , " small " )
def _compute_fileurls ( ignored ) :
self . fileurls = { }
for which in self . uris :
self . fileurls [ which ] = " uri/ " + urllib . quote ( self . uris [ which ] )
d . addCallback ( _compute_fileurls )
def _clobber_shares ( ignored ) :
good_shares = self . find_shares ( self . uris [ " good " ] )
self . failUnlessEqual ( len ( good_shares ) , 10 )
sick_shares = self . find_shares ( self . uris [ " sick " ] )
os . unlink ( sick_shares [ 0 ] [ 2 ] )
dead_shares = self . find_shares ( self . uris [ " dead " ] )
for i in range ( 1 , 10 ) :
os . unlink ( dead_shares [ i ] [ 2 ] )
c_shares = self . find_shares ( self . uris [ " corrupt " ] )
cso = CorruptShareOptions ( )
cso . stdout = StringIO ( )
cso . parseOptions ( [ c_shares [ 0 ] [ 2 ] ] )
corrupt_share ( cso )
d . addCallback ( _clobber_shares )
2009-02-17 06:00:34 +00:00
d . addCallback ( self . CHECK , " good " , " t=check " )
2009-02-17 05:12:42 +00:00
def _got_html_good ( res ) :
self . failUnless ( " Healthy " in res , res )
self . failIf ( " Not Healthy " in res , res )
d . addCallback ( _got_html_good )
2009-02-17 06:00:34 +00:00
d . addCallback ( self . CHECK , " good " , " t=check&return_to=somewhere " )
2009-02-17 05:12:42 +00:00
def _got_html_good_return_to ( res ) :
self . failUnless ( " Healthy " in res , res )
self . failIf ( " Not Healthy " in res , res )
2009-04-07 18:54:59 +00:00
self . failUnless ( ' <a href= " somewhere " >Return to file '
2009-02-17 05:12:42 +00:00
in res , res )
d . addCallback ( _got_html_good_return_to )
2009-02-17 06:00:34 +00:00
d . addCallback ( self . CHECK , " good " , " t=check&output=json " )
2009-02-17 05:12:42 +00:00
def _got_json_good ( res ) :
r = simplejson . loads ( res )
self . failUnlessEqual ( r [ " summary " ] , " Healthy " )
self . failUnless ( r [ " results " ] [ " healthy " ] )
self . failIf ( r [ " results " ] [ " needs-rebalancing " ] )
self . failUnless ( r [ " results " ] [ " recoverable " ] )
d . addCallback ( _got_json_good )
2009-02-17 06:00:34 +00:00
d . addCallback ( self . CHECK , " small " , " t=check " )
2009-02-17 05:12:42 +00:00
def _got_html_small ( res ) :
self . failUnless ( " Literal files are always healthy " in res , res )
self . failIf ( " Not Healthy " in res , res )
d . addCallback ( _got_html_small )
2009-02-17 06:00:34 +00:00
d . addCallback ( self . CHECK , " small " , " t=check&return_to=somewhere " )
2009-02-17 05:12:42 +00:00
def _got_html_small_return_to ( res ) :
self . failUnless ( " Literal files are always healthy " in res , res )
self . failIf ( " Not Healthy " in res , res )
2009-04-07 18:54:59 +00:00
self . failUnless ( ' <a href= " somewhere " >Return to file '
2009-02-17 05:12:42 +00:00
in res , res )
d . addCallback ( _got_html_small_return_to )
2009-02-17 06:00:34 +00:00
d . addCallback ( self . CHECK , " small " , " t=check&output=json " )
2009-02-17 05:12:42 +00:00
def _got_json_small ( res ) :
r = simplejson . loads ( res )
self . failUnlessEqual ( r [ " storage-index " ] , " " )
self . failUnless ( r [ " results " ] [ " healthy " ] )
d . addCallback ( _got_json_small )
2009-02-17 06:00:34 +00:00
d . addCallback ( self . CHECK , " sick " , " t=check " )
2009-02-17 05:12:42 +00:00
def _got_html_sick ( res ) :
self . failUnless ( " Not Healthy " in res , res )
d . addCallback ( _got_html_sick )
2009-02-17 06:00:34 +00:00
d . addCallback ( self . CHECK , " sick " , " t=check&output=json " )
2009-02-17 05:12:42 +00:00
def _got_json_sick ( res ) :
r = simplejson . loads ( res )
self . failUnlessEqual ( r [ " summary " ] ,
" Not Healthy: 9 shares (enc 3-of-10) " )
self . failIf ( r [ " results " ] [ " healthy " ] )
self . failIf ( r [ " results " ] [ " needs-rebalancing " ] )
self . failUnless ( r [ " results " ] [ " recoverable " ] )
d . addCallback ( _got_json_sick )
2009-02-17 06:00:34 +00:00
d . addCallback ( self . CHECK , " dead " , " t=check " )
2009-02-17 05:12:42 +00:00
def _got_html_dead ( res ) :
self . failUnless ( " Not Healthy " in res , res )
d . addCallback ( _got_html_dead )
2009-02-17 06:00:34 +00:00
d . addCallback ( self . CHECK , " dead " , " t=check&output=json " )
2009-02-17 05:12:42 +00:00
def _got_json_dead ( res ) :
r = simplejson . loads ( res )
self . failUnlessEqual ( r [ " summary " ] ,
" Not Healthy: 1 shares (enc 3-of-10) " )
self . failIf ( r [ " results " ] [ " healthy " ] )
self . failIf ( r [ " results " ] [ " needs-rebalancing " ] )
self . failIf ( r [ " results " ] [ " recoverable " ] )
d . addCallback ( _got_json_dead )
2009-02-17 06:00:34 +00:00
d . addCallback ( self . CHECK , " corrupt " , " t=check&verify=true " )
2009-02-17 05:12:42 +00:00
def _got_html_corrupt ( res ) :
self . failUnless ( " Not Healthy! : Unhealthy " in res , res )
d . addCallback ( _got_html_corrupt )
2009-02-17 06:00:34 +00:00
d . addCallback ( self . CHECK , " corrupt " , " t=check&verify=true&output=json " )
2009-02-17 05:12:42 +00:00
def _got_json_corrupt ( res ) :
r = simplejson . loads ( res )
self . failUnless ( " Unhealthy: 9 shares (enc 3-of-10) " in r [ " summary " ] ,
r [ " summary " ] )
self . failIf ( r [ " results " ] [ " healthy " ] )
self . failUnless ( r [ " results " ] [ " recoverable " ] )
self . failUnlessEqual ( r [ " results " ] [ " count-shares-good " ] , 9 )
self . failUnlessEqual ( r [ " results " ] [ " count-corrupt-shares " ] , 1 )
d . addCallback ( _got_json_corrupt )
d . addErrback ( self . explain_web_error )
return d
def test_repair_html ( self ) :
self . basedir = " web/Grid/repair_html "
self . set_up_grid ( )
c0 = self . g . clients [ 0 ]
self . uris = { }
DATA = " data " * 100
d = c0 . upload ( upload . Data ( DATA , convergence = " " ) )
def _stash_uri ( ur , which ) :
self . uris [ which ] = ur . uri
d . addCallback ( _stash_uri , " good " )
d . addCallback ( lambda ign :
c0 . upload ( upload . Data ( DATA + " 1 " , convergence = " " ) ) )
d . addCallback ( _stash_uri , " sick " )
d . addCallback ( lambda ign :
c0 . upload ( upload . Data ( DATA + " 2 " , convergence = " " ) ) )
d . addCallback ( _stash_uri , " dead " )
def _stash_mutable_uri ( n , which ) :
self . uris [ which ] = n . get_uri ( )
assert isinstance ( self . uris [ which ] , str )
d . addCallback ( lambda ign : c0 . create_mutable_file ( DATA + " 3 " ) )
d . addCallback ( _stash_mutable_uri , " corrupt " )
def _compute_fileurls ( ignored ) :
self . fileurls = { }
for which in self . uris :
self . fileurls [ which ] = " uri/ " + urllib . quote ( self . uris [ which ] )
d . addCallback ( _compute_fileurls )
def _clobber_shares ( ignored ) :
good_shares = self . find_shares ( self . uris [ " good " ] )
self . failUnlessEqual ( len ( good_shares ) , 10 )
sick_shares = self . find_shares ( self . uris [ " sick " ] )
os . unlink ( sick_shares [ 0 ] [ 2 ] )
dead_shares = self . find_shares ( self . uris [ " dead " ] )
for i in range ( 1 , 10 ) :
os . unlink ( dead_shares [ i ] [ 2 ] )
c_shares = self . find_shares ( self . uris [ " corrupt " ] )
cso = CorruptShareOptions ( )
cso . stdout = StringIO ( )
cso . parseOptions ( [ c_shares [ 0 ] [ 2 ] ] )
corrupt_share ( cso )
d . addCallback ( _clobber_shares )
2009-02-17 06:00:34 +00:00
d . addCallback ( self . CHECK , " good " , " t=check&repair=true " )
2009-02-17 05:12:42 +00:00
def _got_html_good ( res ) :
self . failUnless ( " Healthy " in res , res )
self . failIf ( " Not Healthy " in res , res )
self . failUnless ( " No repair necessary " in res , res )
d . addCallback ( _got_html_good )
2009-02-17 06:00:34 +00:00
d . addCallback ( self . CHECK , " sick " , " t=check&repair=true " )
2009-02-17 05:12:42 +00:00
def _got_html_sick ( res ) :
self . failUnless ( " Healthy : healthy " in res , res )
self . failIf ( " Not Healthy " in res , res )
self . failUnless ( " Repair successful " in res , res )
d . addCallback ( _got_html_sick )
# repair of a dead file will fail, of course, but it isn't yet
# clear how this should be reported. Right now it shows up as
# a "410 Gone".
#
2009-02-17 06:00:34 +00:00
#d.addCallback(self.CHECK, "dead", "t=check&repair=true")
2009-02-17 05:12:42 +00:00
#def _got_html_dead(res):
# print res
# self.failUnless("Healthy : healthy" in res, res)
# self.failIf("Not Healthy" in res, res)
# self.failUnless("No repair necessary" in res, res)
#d.addCallback(_got_html_dead)
2009-02-17 06:00:34 +00:00
d . addCallback ( self . CHECK , " corrupt " , " t=check&verify=true&repair=true " )
2009-02-17 05:12:42 +00:00
def _got_html_corrupt ( res ) :
self . failUnless ( " Healthy : Healthy " in res , res )
self . failIf ( " Not Healthy " in res , res )
self . failUnless ( " Repair successful " in res , res )
d . addCallback ( _got_html_corrupt )
d . addErrback ( self . explain_web_error )
return d
def test_repair_json ( self ) :
self . basedir = " web/Grid/repair_json "
self . set_up_grid ( )
c0 = self . g . clients [ 0 ]
self . uris = { }
DATA = " data " * 100
d = c0 . upload ( upload . Data ( DATA + " 1 " , convergence = " " ) )
def _stash_uri ( ur , which ) :
self . uris [ which ] = ur . uri
d . addCallback ( _stash_uri , " sick " )
def _compute_fileurls ( ignored ) :
self . fileurls = { }
for which in self . uris :
self . fileurls [ which ] = " uri/ " + urllib . quote ( self . uris [ which ] )
d . addCallback ( _compute_fileurls )
def _clobber_shares ( ignored ) :
sick_shares = self . find_shares ( self . uris [ " sick " ] )
os . unlink ( sick_shares [ 0 ] [ 2 ] )
d . addCallback ( _clobber_shares )
2009-02-17 06:00:34 +00:00
d . addCallback ( self . CHECK , " sick " , " t=check&repair=true&output=json " )
2009-02-17 05:12:42 +00:00
def _got_json_sick ( res ) :
r = simplejson . loads ( res )
self . failUnlessEqual ( r [ " repair-attempted " ] , True )
self . failUnlessEqual ( r [ " repair-successful " ] , True )
self . failUnlessEqual ( r [ " pre-repair-results " ] [ " summary " ] ,
" Not Healthy: 9 shares (enc 3-of-10) " )
self . failIf ( r [ " pre-repair-results " ] [ " results " ] [ " healthy " ] )
self . failUnlessEqual ( r [ " post-repair-results " ] [ " summary " ] , " healthy " )
self . failUnless ( r [ " post-repair-results " ] [ " results " ] [ " healthy " ] )
d . addCallback ( _got_json_sick )
d . addErrback ( self . explain_web_error )
return d
2009-02-17 06:35:53 +00:00
def test_deep_check ( self ) :
self . basedir = " web/Grid/deep_check "
self . set_up_grid ( )
c0 = self . g . clients [ 0 ]
self . uris = { }
self . fileurls = { }
DATA = " data " * 100
d = c0 . create_empty_dirnode ( )
def _stash_root_and_create_file ( n ) :
self . rootnode = n
self . fileurls [ " root " ] = " uri/ " + urllib . quote ( n . get_uri ( ) ) + " / "
return n . add_file ( u " good " , upload . Data ( DATA , convergence = " " ) )
d . addCallback ( _stash_root_and_create_file )
def _stash_uri ( fn , which ) :
self . uris [ which ] = fn . get_uri ( )
2009-02-24 22:40:17 +00:00
return fn
2009-02-17 06:35:53 +00:00
d . addCallback ( _stash_uri , " good " )
d . addCallback ( lambda ign :
self . rootnode . add_file ( u " small " ,
upload . Data ( " literal " ,
convergence = " " ) ) )
d . addCallback ( _stash_uri , " small " )
2009-02-24 22:40:17 +00:00
d . addCallback ( lambda ign :
self . rootnode . add_file ( u " sick " ,
upload . Data ( DATA + " 1 " ,
convergence = " " ) ) )
d . addCallback ( _stash_uri , " sick " )
def _clobber_shares ( ignored ) :
self . delete_shares_numbered ( self . uris [ " sick " ] , [ 0 , 1 ] )
d . addCallback ( _clobber_shares )
# root
# root/good
# root/small
# root/sick
2009-02-17 06:35:53 +00:00
d . addCallback ( self . CHECK , " root " , " t=stream-deep-check " )
def _done ( res ) :
units = [ simplejson . loads ( line )
for line in res . splitlines ( )
if line ]
2009-02-24 22:40:17 +00:00
self . failUnlessEqual ( len ( units ) , 4 + 1 )
2009-02-17 06:35:53 +00:00
# should be parent-first
u0 = units [ 0 ]
self . failUnlessEqual ( u0 [ " path " ] , [ ] )
self . failUnlessEqual ( u0 [ " type " ] , " directory " )
self . failUnlessEqual ( u0 [ " cap " ] , self . rootnode . get_uri ( ) )
u0cr = u0 [ " check-results " ]
self . failUnlessEqual ( u0cr [ " results " ] [ " count-shares-good " ] , 10 )
ugood = [ u for u in units
if u [ " type " ] == " file " and u [ " path " ] == [ u " good " ] ] [ 0 ]
self . failUnlessEqual ( ugood [ " cap " ] , self . uris [ " good " ] )
ugoodcr = ugood [ " check-results " ]
self . failUnlessEqual ( ugoodcr [ " results " ] [ " count-shares-good " ] , 10 )
stats = units [ - 1 ]
self . failUnlessEqual ( stats [ " type " ] , " stats " )
s = stats [ " stats " ]
2009-02-24 22:40:17 +00:00
self . failUnlessEqual ( s [ " count-immutable-files " ] , 2 )
2009-02-17 06:35:53 +00:00
self . failUnlessEqual ( s [ " count-literal-files " ] , 1 )
self . failUnlessEqual ( s [ " count-directories " ] , 1 )
d . addCallback ( _done )
2009-02-24 22:40:17 +00:00
# now add root/subdir and root/subdir/grandchild, then make subdir
# unrecoverable, then see what happens
d . addCallback ( lambda ign :
self . rootnode . create_empty_directory ( u " subdir " ) )
d . addCallback ( _stash_uri , " subdir " )
d . addCallback ( lambda subdir_node :
subdir_node . add_file ( u " grandchild " ,
upload . Data ( DATA + " 2 " ,
convergence = " " ) ) )
d . addCallback ( _stash_uri , " grandchild " )
d . addCallback ( lambda ign :
self . delete_shares_numbered ( self . uris [ " subdir " ] ,
2009-02-25 06:13:35 +00:00
range ( 1 , 10 ) ) )
# root
# root/good
# root/small
# root/sick
# root/subdir [unrecoverable]
# root/subdir/grandchild
# how should a streaming-JSON API indicate fatal error?
# answer: emit ERROR: instead of a JSON string
d . addCallback ( self . CHECK , " root " , " t=stream-manifest " )
def _check_broken_manifest ( res ) :
lines = res . splitlines ( )
error_lines = [ i
for ( i , line ) in enumerate ( lines )
if line . startswith ( " ERROR: " ) ]
if not error_lines :
self . fail ( " no ERROR: in output: %s " % ( res , ) )
first_error = error_lines [ 0 ]
error_line = lines [ first_error ]
error_msg = lines [ first_error + 1 : ]
error_msg_s = " \n " . join ( error_msg ) + " \n "
2009-02-25 08:46:21 +00:00
self . failUnlessIn ( " ERROR: UnrecoverableFileError(no recoverable versions) " ,
error_line )
2009-02-25 06:13:35 +00:00
self . failUnless ( len ( error_msg ) > 2 , error_msg_s ) # some traceback
units = [ simplejson . loads ( line ) for line in lines [ : first_error ] ]
self . failUnlessEqual ( len ( units ) , 5 ) # includes subdir
last_unit = units [ - 1 ]
self . failUnlessEqual ( last_unit [ " path " ] , [ " subdir " ] )
d . addCallback ( _check_broken_manifest )
d . addCallback ( self . CHECK , " root " , " t=stream-deep-check " )
def _check_broken_deepcheck ( res ) :
lines = res . splitlines ( )
error_lines = [ i
for ( i , line ) in enumerate ( lines )
if line . startswith ( " ERROR: " ) ]
if not error_lines :
self . fail ( " no ERROR: in output: %s " % ( res , ) )
first_error = error_lines [ 0 ]
error_line = lines [ first_error ]
error_msg = lines [ first_error + 1 : ]
error_msg_s = " \n " . join ( error_msg ) + " \n "
2009-02-25 08:46:21 +00:00
self . failUnlessIn ( " ERROR: UnrecoverableFileError(no recoverable versions) " ,
error_line )
2009-02-25 06:13:35 +00:00
self . failUnless ( len ( error_msg ) > 2 , error_msg_s ) # some traceback
units = [ simplejson . loads ( line ) for line in lines [ : first_error ] ]
self . failUnlessEqual ( len ( units ) , 5 ) # includes subdir
last_unit = units [ - 1 ]
self . failUnlessEqual ( last_unit [ " path " ] , [ " subdir " ] )
r = last_unit [ " check-results " ] [ " results " ]
self . failUnlessEqual ( r [ " count-recoverable-versions " ] , 0 )
self . failUnlessEqual ( r [ " count-shares-good " ] , 1 )
self . failUnlessEqual ( r [ " recoverable " ] , False )
d . addCallback ( _check_broken_deepcheck )
2009-02-24 22:40:17 +00:00
2009-02-17 06:35:53 +00:00
d . addErrback ( self . explain_web_error )
return d
def test_deep_check_and_repair ( self ) :
self . basedir = " web/Grid/deep_check_and_repair "
self . set_up_grid ( )
c0 = self . g . clients [ 0 ]
self . uris = { }
self . fileurls = { }
DATA = " data " * 100
d = c0 . create_empty_dirnode ( )
def _stash_root_and_create_file ( n ) :
self . rootnode = n
self . fileurls [ " root " ] = " uri/ " + urllib . quote ( n . get_uri ( ) ) + " / "
return n . add_file ( u " good " , upload . Data ( DATA , convergence = " " ) )
d . addCallback ( _stash_root_and_create_file )
def _stash_uri ( fn , which ) :
self . uris [ which ] = fn . get_uri ( )
d . addCallback ( _stash_uri , " good " )
d . addCallback ( lambda ign :
self . rootnode . add_file ( u " small " ,
upload . Data ( " literal " ,
convergence = " " ) ) )
d . addCallback ( _stash_uri , " small " )
d . addCallback ( lambda ign :
self . rootnode . add_file ( u " sick " ,
upload . Data ( DATA + " 1 " ,
convergence = " " ) ) )
d . addCallback ( _stash_uri , " sick " )
#d.addCallback(lambda ign:
# self.rootnode.add_file(u"dead",
# upload.Data(DATA+"2",
# convergence="")))
#d.addCallback(_stash_uri, "dead")
#d.addCallback(lambda ign: c0.create_mutable_file("mutable"))
#d.addCallback(lambda fn: self.rootnode.set_node(u"corrupt", fn))
#d.addCallback(_stash_uri, "corrupt")
def _clobber_shares ( ignored ) :
good_shares = self . find_shares ( self . uris [ " good " ] )
self . failUnlessEqual ( len ( good_shares ) , 10 )
sick_shares = self . find_shares ( self . uris [ " sick " ] )
os . unlink ( sick_shares [ 0 ] [ 2 ] )
#dead_shares = self.find_shares(self.uris["dead"])
#for i in range(1, 10):
# os.unlink(dead_shares[i][2])
#c_shares = self.find_shares(self.uris["corrupt"])
#cso = CorruptShareOptions()
#cso.stdout = StringIO()
#cso.parseOptions([c_shares[0][2]])
#corrupt_share(cso)
d . addCallback ( _clobber_shares )
2009-02-24 22:40:17 +00:00
# root
# root/good CHK, 10 shares
# root/small LIT
# root/sick CHK, 9 shares
2009-02-17 06:35:53 +00:00
d . addCallback ( self . CHECK , " root " , " t=stream-deep-check&repair=true " )
def _done ( res ) :
units = [ simplejson . loads ( line )
for line in res . splitlines ( )
if line ]
self . failUnlessEqual ( len ( units ) , 4 + 1 )
# should be parent-first
u0 = units [ 0 ]
self . failUnlessEqual ( u0 [ " path " ] , [ ] )
self . failUnlessEqual ( u0 [ " type " ] , " directory " )
self . failUnlessEqual ( u0 [ " cap " ] , self . rootnode . get_uri ( ) )
u0crr = u0 [ " check-and-repair-results " ]
self . failUnlessEqual ( u0crr [ " repair-attempted " ] , False )
self . failUnlessEqual ( u0crr [ " pre-repair-results " ] [ " results " ] [ " count-shares-good " ] , 10 )
ugood = [ u for u in units
if u [ " type " ] == " file " and u [ " path " ] == [ u " good " ] ] [ 0 ]
self . failUnlessEqual ( ugood [ " cap " ] , self . uris [ " good " ] )
ugoodcrr = ugood [ " check-and-repair-results " ]
self . failUnlessEqual ( u0crr [ " repair-attempted " ] , False )
self . failUnlessEqual ( u0crr [ " pre-repair-results " ] [ " results " ] [ " count-shares-good " ] , 10 )
usick = [ u for u in units
if u [ " type " ] == " file " and u [ " path " ] == [ u " sick " ] ] [ 0 ]
self . failUnlessEqual ( usick [ " cap " ] , self . uris [ " sick " ] )
usickcrr = usick [ " check-and-repair-results " ]
self . failUnlessEqual ( usickcrr [ " repair-attempted " ] , True )
self . failUnlessEqual ( usickcrr [ " repair-successful " ] , True )
self . failUnlessEqual ( usickcrr [ " pre-repair-results " ] [ " results " ] [ " count-shares-good " ] , 9 )
self . failUnlessEqual ( usickcrr [ " post-repair-results " ] [ " results " ] [ " count-shares-good " ] , 10 )
stats = units [ - 1 ]
self . failUnlessEqual ( stats [ " type " ] , " stats " )
s = stats [ " stats " ]
self . failUnlessEqual ( s [ " count-immutable-files " ] , 2 )
self . failUnlessEqual ( s [ " count-literal-files " ] , 1 )
self . failUnlessEqual ( s [ " count-directories " ] , 1 )
d . addCallback ( _done )
d . addErrback ( self . explain_web_error )
return d
2009-02-18 02:32:43 +00:00
def _count_leases ( self , ignored , which ) :
u = self . uris [ which ]
shares = self . find_shares ( u )
lease_counts = [ ]
for shnum , serverid , fn in shares :
2009-03-07 05:45:17 +00:00
sf = get_share_file ( fn )
num_leases = len ( list ( sf . get_leases ( ) ) )
2009-02-18 02:32:43 +00:00
lease_counts . append ( ( fn , num_leases ) )
return lease_counts
def _assert_leasecount ( self , lease_counts , expected ) :
for ( fn , num_leases ) in lease_counts :
if num_leases != expected :
self . fail ( " expected %d leases, have %d , on %s " %
( expected , num_leases , fn ) )
def test_add_lease ( self ) :
self . basedir = " web/Grid/add_lease "
self . set_up_grid ( num_clients = 2 )
c0 = self . g . clients [ 0 ]
self . uris = { }
DATA = " data " * 100
d = c0 . upload ( upload . Data ( DATA , convergence = " " ) )
def _stash_uri ( ur , which ) :
self . uris [ which ] = ur . uri
d . addCallback ( _stash_uri , " one " )
d . addCallback ( lambda ign :
c0 . upload ( upload . Data ( DATA + " 1 " , convergence = " " ) ) )
d . addCallback ( _stash_uri , " two " )
def _stash_mutable_uri ( n , which ) :
self . uris [ which ] = n . get_uri ( )
assert isinstance ( self . uris [ which ] , str )
d . addCallback ( lambda ign : c0 . create_mutable_file ( DATA + " 2 " ) )
d . addCallback ( _stash_mutable_uri , " mutable " )
def _compute_fileurls ( ignored ) :
self . fileurls = { }
for which in self . uris :
self . fileurls [ which ] = " uri/ " + urllib . quote ( self . uris [ which ] )
d . addCallback ( _compute_fileurls )
d . addCallback ( self . _count_leases , " one " )
d . addCallback ( self . _assert_leasecount , 1 )
d . addCallback ( self . _count_leases , " two " )
d . addCallback ( self . _assert_leasecount , 1 )
d . addCallback ( self . _count_leases , " mutable " )
d . addCallback ( self . _assert_leasecount , 1 )
d . addCallback ( self . CHECK , " one " , " t=check " ) # no add-lease
def _got_html_good ( res ) :
self . failUnless ( " Healthy " in res , res )
self . failIf ( " Not Healthy " in res , res )
d . addCallback ( _got_html_good )
d . addCallback ( self . _count_leases , " one " )
d . addCallback ( self . _assert_leasecount , 1 )
d . addCallback ( self . _count_leases , " two " )
d . addCallback ( self . _assert_leasecount , 1 )
d . addCallback ( self . _count_leases , " mutable " )
d . addCallback ( self . _assert_leasecount , 1 )
# this CHECK uses the original client, which uses the same
# lease-secrets, so it will just renew the original lease
d . addCallback ( self . CHECK , " one " , " t=check&add-lease=true " )
d . addCallback ( _got_html_good )
d . addCallback ( self . _count_leases , " one " )
d . addCallback ( self . _assert_leasecount , 1 )
d . addCallback ( self . _count_leases , " two " )
d . addCallback ( self . _assert_leasecount , 1 )
d . addCallback ( self . _count_leases , " mutable " )
d . addCallback ( self . _assert_leasecount , 1 )
# this CHECK uses an alternate client, which adds a second lease
d . addCallback ( self . CHECK , " one " , " t=check&add-lease=true " , clientnum = 1 )
d . addCallback ( _got_html_good )
d . addCallback ( self . _count_leases , " one " )
d . addCallback ( self . _assert_leasecount , 2 )
d . addCallback ( self . _count_leases , " two " )
d . addCallback ( self . _assert_leasecount , 1 )
d . addCallback ( self . _count_leases , " mutable " )
d . addCallback ( self . _assert_leasecount , 1 )
d . addCallback ( self . CHECK , " mutable " , " t=check&add-lease=true " )
d . addCallback ( _got_html_good )
d . addCallback ( self . _count_leases , " one " )
d . addCallback ( self . _assert_leasecount , 2 )
d . addCallback ( self . _count_leases , " two " )
d . addCallback ( self . _assert_leasecount , 1 )
d . addCallback ( self . _count_leases , " mutable " )
d . addCallback ( self . _assert_leasecount , 1 )
d . addCallback ( self . CHECK , " mutable " , " t=check&add-lease=true " ,
clientnum = 1 )
d . addCallback ( _got_html_good )
d . addCallback ( self . _count_leases , " one " )
d . addCallback ( self . _assert_leasecount , 2 )
d . addCallback ( self . _count_leases , " two " )
d . addCallback ( self . _assert_leasecount , 1 )
d . addCallback ( self . _count_leases , " mutable " )
d . addCallback ( self . _assert_leasecount , 2 )
d . addErrback ( self . explain_web_error )
return d
def test_deep_add_lease ( self ) :
self . basedir = " web/Grid/deep_add_lease "
self . set_up_grid ( num_clients = 2 )
c0 = self . g . clients [ 0 ]
self . uris = { }
self . fileurls = { }
DATA = " data " * 100
d = c0 . create_empty_dirnode ( )
def _stash_root_and_create_file ( n ) :
self . rootnode = n
self . uris [ " root " ] = n . get_uri ( )
self . fileurls [ " root " ] = " uri/ " + urllib . quote ( n . get_uri ( ) ) + " / "
return n . add_file ( u " one " , upload . Data ( DATA , convergence = " " ) )
d . addCallback ( _stash_root_and_create_file )
def _stash_uri ( fn , which ) :
self . uris [ which ] = fn . get_uri ( )
d . addCallback ( _stash_uri , " one " )
d . addCallback ( lambda ign :
self . rootnode . add_file ( u " small " ,
upload . Data ( " literal " ,
convergence = " " ) ) )
d . addCallback ( _stash_uri , " small " )
d . addCallback ( lambda ign : c0 . create_mutable_file ( " mutable " ) )
d . addCallback ( lambda fn : self . rootnode . set_node ( u " mutable " , fn ) )
d . addCallback ( _stash_uri , " mutable " )
d . addCallback ( self . CHECK , " root " , " t=stream-deep-check " ) # no add-lease
def _done ( res ) :
units = [ simplejson . loads ( line )
for line in res . splitlines ( )
if line ]
# root, one, small, mutable, stats
self . failUnlessEqual ( len ( units ) , 4 + 1 )
d . addCallback ( _done )
d . addCallback ( self . _count_leases , " root " )
d . addCallback ( self . _assert_leasecount , 1 )
d . addCallback ( self . _count_leases , " one " )
d . addCallback ( self . _assert_leasecount , 1 )
d . addCallback ( self . _count_leases , " mutable " )
d . addCallback ( self . _assert_leasecount , 1 )
d . addCallback ( self . CHECK , " root " , " t=stream-deep-check&add-lease=true " )
d . addCallback ( _done )
d . addCallback ( self . _count_leases , " root " )
d . addCallback ( self . _assert_leasecount , 1 )
d . addCallback ( self . _count_leases , " one " )
d . addCallback ( self . _assert_leasecount , 1 )
d . addCallback ( self . _count_leases , " mutable " )
d . addCallback ( self . _assert_leasecount , 1 )
d . addCallback ( self . CHECK , " root " , " t=stream-deep-check&add-lease=true " ,
clientnum = 1 )
d . addCallback ( _done )
d . addCallback ( self . _count_leases , " root " )
d . addCallback ( self . _assert_leasecount , 2 )
d . addCallback ( self . _count_leases , " one " )
d . addCallback ( self . _assert_leasecount , 2 )
d . addCallback ( self . _count_leases , " mutable " )
d . addCallback ( self . _assert_leasecount , 2 )
d . addErrback ( self . explain_web_error )
return d
2009-03-04 04:56:30 +00:00
def test_exceptions ( self ) :
self . basedir = " web/Grid/exceptions "
self . set_up_grid ( num_clients = 1 , num_servers = 2 )
c0 = self . g . clients [ 0 ]
self . fileurls = { }
DATA = " data " * 100
d = c0 . create_empty_dirnode ( )
def _stash_root ( n ) :
self . fileurls [ " root " ] = " uri/ " + urllib . quote ( n . get_uri ( ) ) + " / "
self . fileurls [ " imaginary " ] = self . fileurls [ " root " ] + " imaginary "
return n
d . addCallback ( _stash_root )
d . addCallback ( lambda ign : c0 . upload ( upload . Data ( DATA , convergence = " " ) ) )
def _stash_bad ( ur ) :
self . fileurls [ " 1share " ] = " uri/ " + urllib . quote ( ur . uri )
self . delete_shares_numbered ( ur . uri , range ( 1 , 10 ) )
u = uri . from_string ( ur . uri )
u . key = testutil . flip_bit ( u . key , 0 )
baduri = u . to_string ( )
self . fileurls [ " 0shares " ] = " uri/ " + urllib . quote ( baduri )
d . addCallback ( _stash_bad )
2009-03-07 11:57:07 +00:00
d . addCallback ( lambda ign : c0 . create_empty_dirnode ( ) )
def _mangle_dirnode_1share ( n ) :
u = n . get_uri ( )
url = self . fileurls [ " dir-1share " ] = " uri/ " + urllib . quote ( u ) + " / "
self . fileurls [ " dir-1share-json " ] = url + " ?t=json "
self . delete_shares_numbered ( u , range ( 1 , 10 ) )
d . addCallback ( _mangle_dirnode_1share )
d . addCallback ( lambda ign : c0 . create_empty_dirnode ( ) )
def _mangle_dirnode_0share ( n ) :
u = n . get_uri ( )
url = self . fileurls [ " dir-0share " ] = " uri/ " + urllib . quote ( u ) + " / "
self . fileurls [ " dir-0share-json " ] = url + " ?t=json "
self . delete_shares_numbered ( u , range ( 0 , 10 ) )
d . addCallback ( _mangle_dirnode_0share )
2009-03-04 04:56:30 +00:00
# NotEnoughSharesError should be reported sensibly, with a
# text/plain explanation of the problem, and perhaps some
# information on which shares *could* be found.
d . addCallback ( lambda ignored :
self . shouldHTTPError ( " GET unrecoverable " ,
2009-06-25 02:17:07 +00:00
410 , " Gone " , " NoSharesError " ,
2009-03-04 04:56:30 +00:00
self . GET , self . fileurls [ " 0shares " ] ) )
def _check_zero_shares ( body ) :
self . failIf ( " <html> " in body , body )
body = " " . join ( body . strip ( ) . split ( ) )
2009-06-25 02:17:07 +00:00
exp = ( " NoSharesError: no shares could be found. "
2009-03-04 04:56:30 +00:00
" Zero shares usually indicates a corrupt URI, or that "
" no servers were connected, but it might also indicate "
" severe corruption. You should perform a filecheck on "
2009-06-25 02:17:07 +00:00
" this object to learn more. The full error message is: "
" Failed to get enough shareholders: have 0, need 3 " )
2009-03-04 04:56:30 +00:00
self . failUnlessEqual ( exp , body )
d . addCallback ( _check_zero_shares )
2009-03-07 11:57:07 +00:00
2009-03-04 04:56:30 +00:00
d . addCallback ( lambda ignored :
self . shouldHTTPError ( " GET 1share " ,
410 , " Gone " , " NotEnoughSharesError " ,
self . GET , self . fileurls [ " 1share " ] ) )
def _check_one_share ( body ) :
self . failIf ( " <html> " in body , body )
body = " " . join ( body . strip ( ) . split ( ) )
2009-06-25 02:17:07 +00:00
exp = ( " NotEnoughSharesError: This indicates that some "
2009-03-04 04:56:30 +00:00
" servers were unavailable, or that shares have been "
" lost to server departure, hard drive failure, or disk "
" corruption. You should perform a filecheck on "
2009-06-25 02:17:07 +00:00
" this object to learn more. The full error message is: "
" Failed to get enough shareholders: have 1, need 3 " )
2009-03-04 04:56:30 +00:00
self . failUnlessEqual ( exp , body )
d . addCallback ( _check_one_share )
d . addCallback ( lambda ignored :
self . shouldHTTPError ( " GET imaginary " ,
404 , " Not Found " , None ,
self . GET , self . fileurls [ " imaginary " ] ) )
def _missing_child ( body ) :
self . failUnless ( " No such child: imaginary " in body , body )
d . addCallback ( _missing_child )
2009-03-07 11:57:07 +00:00
d . addCallback ( lambda ignored : self . GET ( self . fileurls [ " dir-0share " ] ) )
def _check_0shares_dir_html ( body ) :
self . failUnless ( " <html> " in body , body )
# we should see the regular page, but without the child table or
# the dirops forms
body = " " . join ( body . strip ( ) . split ( ) )
self . failUnlessIn ( ' href= " ?t=info " >More info on this directory ' ,
body )
exp = ( " UnrecoverableFileError: the directory (or mutable file) "
" could not be retrieved, because there were insufficient "
" good shares. This might indicate that no servers were "
" connected, insufficient servers were connected, the URI "
" was corrupt, or that shares have been lost due to server "
" departure, hard drive failure, or disk corruption. You "
" should perform a filecheck on this object to learn more. " )
self . failUnlessIn ( exp , body )
self . failUnlessIn ( " No upload forms: directory is unreadable " , body )
d . addCallback ( _check_0shares_dir_html )
d . addCallback ( lambda ignored : self . GET ( self . fileurls [ " dir-1share " ] ) )
def _check_1shares_dir_html ( body ) :
# at some point, we'll split UnrecoverableFileError into 0-shares
# and some-shares like we did for immutable files (since there
# are different sorts of advice to offer in each case). For now,
# they present the same way.
self . failUnless ( " <html> " in body , body )
body = " " . join ( body . strip ( ) . split ( ) )
self . failUnlessIn ( ' href= " ?t=info " >More info on this directory ' ,
body )
exp = ( " UnrecoverableFileError: the directory (or mutable file) "
" could not be retrieved, because there were insufficient "
" good shares. This might indicate that no servers were "
" connected, insufficient servers were connected, the URI "
" was corrupt, or that shares have been lost due to server "
" departure, hard drive failure, or disk corruption. You "
" should perform a filecheck on this object to learn more. " )
self . failUnlessIn ( exp , body )
self . failUnlessIn ( " No upload forms: directory is unreadable " , body )
d . addCallback ( _check_1shares_dir_html )
d . addCallback ( lambda ignored :
self . shouldHTTPError ( " GET dir-0share-json " ,
410 , " Gone " , " UnrecoverableFileError " ,
self . GET ,
self . fileurls [ " dir-0share-json " ] ) )
def _check_unrecoverable_file ( body ) :
self . failIf ( " <html> " in body , body )
body = " " . join ( body . strip ( ) . split ( ) )
exp = ( " UnrecoverableFileError: the directory (or mutable file) "
" could not be retrieved, because there were insufficient "
" good shares. This might indicate that no servers were "
" connected, insufficient servers were connected, the URI "
" was corrupt, or that shares have been lost due to server "
" departure, hard drive failure, or disk corruption. You "
" should perform a filecheck on this object to learn more. " )
self . failUnlessEqual ( exp , body )
d . addCallback ( _check_unrecoverable_file )
d . addCallback ( lambda ignored :
self . shouldHTTPError ( " GET dir-1share-json " ,
410 , " Gone " , " UnrecoverableFileError " ,
self . GET ,
self . fileurls [ " dir-1share-json " ] ) )
d . addCallback ( _check_unrecoverable_file )
d . addCallback ( lambda ignored :
self . shouldHTTPError ( " GET imaginary " ,
404 , " Not Found " , None ,
self . GET , self . fileurls [ " imaginary " ] ) )
2009-03-04 04:56:30 +00:00
# attach a webapi child that throws a random error, to test how it
# gets rendered.
w = c0 . getServiceNamed ( " webish " )
w . root . putChild ( " ERRORBOOM " , ErrorBoom ( ) )
d . addCallback ( lambda ignored :
self . shouldHTTPError ( " GET errorboom_html " ,
500 , " Internal Server Error " , None ,
self . GET , " ERRORBOOM " ) )
def _internal_error_html ( body ) :
# test that a weird exception during a webapi operation with
# Accept:*/* results in a text/html stack trace, while one
# without that Accept: line gets us a text/plain stack trace
self . failUnless ( " <html> " in body , " expected HTML, not ' %s ' " % body )
d . addCallback ( _internal_error_html )
d . addCallback ( lambda ignored :
self . shouldHTTPError ( " GET errorboom_text " ,
500 , " Internal Server Error " , None ,
self . GET , " ERRORBOOM " ,
headers = { " accept " : [ " text/plain " ] } ) )
def _internal_error_text ( body ) :
# test that a weird exception during a webapi operation with
# Accept:*/* results in a text/html stack trace, while one
# without that Accept: line gets us a text/plain stack trace
self . failIf ( " <html> " in body , body )
self . failUnless ( body . startswith ( " Traceback " ) , body )
d . addCallback ( _internal_error_text )
def _flush_errors ( res ) :
# Trial: please ignore the CompletelyUnhandledError in the logs
self . flushLoggedErrors ( CompletelyUnhandledError )
return res
d . addBoth ( _flush_errors )
return d
class CompletelyUnhandledError ( Exception ) :
pass
class ErrorBoom ( rend . Page ) :
def beforeRender ( self , ctx ) :
raise CompletelyUnhandledError ( " whoops " )