2006-12-03 01:30:18 +00:00
2007-03-29 21:31:55 +00:00
import os
2007-06-26 22:55:00 +00:00
from cStringIO import StringIO
2006-12-03 01:30:18 +00:00
from twisted . trial import unittest
2006-12-03 02:37:31 +00:00
from twisted . internet import defer , reactor
2006-12-03 01:30:18 +00:00
from twisted . application import service
2007-06-15 07:37:32 +00:00
from allmydata import client , uri , download , upload
2007-06-01 01:31:01 +00:00
from allmydata . introducer_and_vdrive import IntroducerAndVdrive
2007-04-24 04:15:02 +00:00
from allmydata . util import idlib , fileutil , testutil
2007-06-26 22:55:00 +00:00
from allmydata . scripts import runner
2007-07-21 22:40:36 +00:00
from allmydata . interfaces import IDirectoryNode , IFileNode , IFileURI
2007-06-28 22:05:33 +00:00
from allmydata . dirnode import NotMutableError
2006-12-03 02:37:31 +00:00
from foolscap . eventual import flushEventualQueue
2006-12-03 03:32:08 +00:00
from twisted . python import log
2007-04-19 01:29:10 +00:00
from twisted . python . failure import Failure
2006-12-07 19:48:06 +00:00
from twisted . web . client import getPage
2007-07-08 03:06:44 +00:00
from twisted . web . error import Error
2006-12-03 01:30:18 +00:00
2007-04-07 03:34:32 +00:00
def flush_but_dont_ignore ( res ) :
d = flushEventualQueue ( )
def _done ( ignored ) :
return res
d . addCallback ( _done )
return d
2007-07-12 20:22:36 +00:00
LARGE_DATA = """
This is some data to publish to the virtual drive , which needs to be large
enough to not fit inside a LIT uri .
"""
2007-04-24 04:15:02 +00:00
class SystemTest ( testutil . SignalMixin , unittest . TestCase ) :
2007-01-10 02:40:36 +00:00
2006-12-03 01:30:18 +00:00
def setUp ( self ) :
self . sparent = service . MultiService ( )
self . sparent . startService ( )
def tearDown ( self ) :
2007-01-10 02:40:36 +00:00
log . msg ( " shutting down SystemTest services " )
2006-12-03 02:37:31 +00:00
d = self . sparent . stopService ( )
2007-04-07 03:34:32 +00:00
d . addBoth ( flush_but_dont_ignore )
2006-12-03 02:37:31 +00:00
return d
2006-12-03 01:30:18 +00:00
2007-04-07 03:34:32 +00:00
def getdir ( self , subdir ) :
return os . path . join ( self . basedir , subdir )
2006-12-03 07:56:27 +00:00
def add_service ( self , s ) :
2006-12-03 01:30:18 +00:00
s . setServiceParent ( self . sparent )
return s
2006-12-03 07:56:27 +00:00
def set_up_nodes ( self , NUMCLIENTS = 5 ) :
self . numclients = NUMCLIENTS
2007-06-01 01:31:01 +00:00
iv_dir = self . getdir ( " introducer_and_vdrive " )
if not os . path . isdir ( iv_dir ) :
fileutil . make_dirs ( iv_dir )
iv = IntroducerAndVdrive ( basedir = iv_dir )
self . introducer_and_vdrive = self . add_service ( iv )
2007-04-30 16:57:52 +00:00
d = self . introducer_and_vdrive . when_tub_ready ( )
2007-03-08 22:10:36 +00:00
d . addCallback ( self . _set_up_nodes_2 )
return d
def _set_up_nodes_2 ( self , res ) :
2007-04-30 16:57:52 +00:00
q = self . introducer_and_vdrive
self . introducer_furl = q . urls [ " introducer " ]
2007-03-27 23:12:11 +00:00
self . vdrive_furl = q . urls [ " vdrive " ]
2006-12-03 07:56:27 +00:00
self . clients = [ ]
2007-03-08 22:10:36 +00:00
for i in range ( self . numclients ) :
2007-04-07 03:34:32 +00:00
basedir = self . getdir ( " client %d " % i )
2006-12-03 03:32:08 +00:00
if not os . path . isdir ( basedir ) :
2007-04-07 03:34:32 +00:00
fileutil . make_dirs ( basedir )
2006-12-07 19:48:06 +00:00
if i == 0 :
2007-03-27 23:12:11 +00:00
open ( os . path . join ( basedir , " webport " ) , " w " ) . write ( " tcp:0:interface=127.0.0.1 " )
2007-04-30 16:57:52 +00:00
open ( os . path . join ( basedir , " introducer.furl " ) , " w " ) . write ( self . introducer_furl )
2007-03-27 23:12:11 +00:00
open ( os . path . join ( basedir , " vdrive.furl " ) , " w " ) . write ( self . vdrive_furl )
2006-12-03 07:56:27 +00:00
c = self . add_service ( client . Client ( basedir = basedir ) )
self . clients . append ( c )
log . msg ( " STARTING " )
2006-12-07 19:48:06 +00:00
d = self . wait_for_connections ( )
def _connected ( res ) :
2007-05-31 19:01:14 +00:00
log . msg ( " CONNECTED " )
2006-12-07 19:48:06 +00:00
# now find out where the web port was
l = self . clients [ 0 ] . getServiceNamed ( " webish " ) . listener
port = l . _port . getHost ( ) . port
self . webish_url = " http://localhost: %d / " % port
d . addCallback ( _connected )
return d
2006-12-03 02:37:31 +00:00
2007-01-10 02:40:36 +00:00
def add_extra_node ( self , client_num ) :
# this node is *not* parented to our self.sparent, so we can shut it
# down separately from the rest, to exercise the connection-lost code
2007-04-07 03:34:32 +00:00
basedir = self . getdir ( " client %d " % client_num )
2007-01-10 02:40:36 +00:00
if not os . path . isdir ( basedir ) :
2007-04-07 03:34:32 +00:00
fileutil . make_dirs ( basedir )
2007-04-30 16:57:52 +00:00
open ( os . path . join ( basedir , " introducer.furl " ) , " w " ) . write ( self . introducer_furl )
2007-03-27 23:12:11 +00:00
open ( os . path . join ( basedir , " vdrive.furl " ) , " w " ) . write ( self . vdrive_furl )
2007-01-10 02:40:36 +00:00
c = client . Client ( basedir = basedir )
self . clients . append ( c )
self . numclients + = 1
c . startService ( )
d = self . wait_for_connections ( )
d . addCallback ( lambda res : c )
return d
2006-12-03 07:56:27 +00:00
def wait_for_connections ( self , ignored = None ) :
for c in self . clients :
2007-03-28 00:44:49 +00:00
if ( not c . introducer_client or
len ( list ( c . get_all_peerids ( ) ) ) != self . numclients ) :
2006-12-03 07:56:27 +00:00
d = defer . Deferred ( )
d . addCallback ( self . wait_for_connections )
reactor . callLater ( 0.05 , d . callback , None )
return d
return defer . succeed ( None )
2006-12-03 02:37:31 +00:00
2006-12-03 03:32:08 +00:00
def test_connections ( self ) :
2007-06-28 18:00:03 +00:00
self . basedir = " system/SystemTest/test_connections "
2006-12-03 07:56:27 +00:00
d = self . set_up_nodes ( )
2007-03-27 23:12:11 +00:00
self . extra_node = None
2007-01-10 02:40:36 +00:00
d . addCallback ( lambda res : self . add_extra_node ( 5 ) )
def _check ( extra_node ) :
2007-01-16 22:12:49 +00:00
self . extra_node = extra_node
2006-12-03 03:32:08 +00:00
for c in self . clients :
2007-03-28 00:44:49 +00:00
self . failUnlessEqual ( len ( list ( c . get_all_peerids ( ) ) ) , 6 )
2006-12-03 03:32:08 +00:00
d . addCallback ( _check )
2007-01-16 22:12:49 +00:00
def _shutdown_extra_node ( res ) :
2007-03-27 23:12:11 +00:00
if self . extra_node :
2007-04-04 23:09:13 +00:00
return self . extra_node . stopService ( )
2007-03-27 23:12:11 +00:00
return res
2007-01-16 22:12:49 +00:00
d . addBoth ( _shutdown_extra_node )
2006-12-03 03:32:08 +00:00
return d
2007-04-07 03:34:32 +00:00
test_connections . timeout = 300
2007-04-18 23:06:57 +00:00
# test_connections is subsumed by test_upload_and_download, and takes
# quite a while to run on a slow machine (because of all the TLS
# connections that must be established). If we ever rework the introducer
# code to such an extent that we're not sure if it works anymore, we can
# reinstate this test until it does.
del test_connections
2006-12-03 03:32:08 +00:00
2006-12-03 10:01:43 +00:00
def test_upload_and_download ( self ) :
2007-06-28 18:00:03 +00:00
self . basedir = " system/SystemTest/test_upload_and_download "
2007-04-17 20:40:47 +00:00
# we use 4000 bytes of data, which will result in about 400k written
# to disk among all our simulated nodes
DATA = " Some data to upload \n " * 200
2006-12-03 07:56:27 +00:00
d = self . set_up_nodes ( )
2006-12-03 10:01:43 +00:00
def _do_upload ( res ) :
2006-12-03 07:56:27 +00:00
log . msg ( " UPLOADING " )
2006-12-03 03:32:08 +00:00
u = self . clients [ 0 ] . getServiceNamed ( " uploader " )
2007-04-19 01:29:10 +00:00
self . uploader = u
2007-04-17 20:40:47 +00:00
# we crank the max segsize down to 1024b for the duration of this
# test, so we can exercise multiple segments. It is important
# that this is not a multiple of the segment size, so that the
2007-04-18 03:28:19 +00:00
# tail segment is not the same length as the others. This actualy
# gets rounded up to 1025 to be a multiple of the number of
# required shares (since we use 25 out of 100 FEC).
2007-04-17 20:40:47 +00:00
options = { " max_segment_size " : 1024 }
d1 = u . upload_data ( DATA , options )
2006-12-03 03:32:08 +00:00
return d1
2006-12-03 10:01:43 +00:00
d . addCallback ( _do_upload )
2007-01-16 04:22:22 +00:00
def _upload_done ( uri ) :
log . msg ( " upload finished: uri is %s " % ( uri , ) )
2007-04-16 20:07:36 +00:00
self . uri = uri
2006-12-03 10:01:43 +00:00
dl = self . clients [ 1 ] . getServiceNamed ( " downloader " )
2007-04-16 20:07:36 +00:00
self . downloader = dl
2006-12-03 10:01:43 +00:00
d . addCallback ( _upload_done )
2007-04-19 01:29:10 +00:00
def _upload_again ( res ) :
# upload again. This ought to be short-circuited, however with
# the way we currently generate URIs (i.e. because they include
# the roothash), we have to do all of the encoding work, and only
# get to save on the upload part.
log . msg ( " UPLOADING AGAIN " )
options = { " max_segment_size " : 1024 }
d1 = self . uploader . upload_data ( DATA , options )
d . addCallback ( _upload_again )
def _download_to_data ( res ) :
log . msg ( " DOWNLOADING " )
return self . downloader . download_to_data ( self . uri )
d . addCallback ( _download_to_data )
def _download_to_data_done ( data ) :
2006-12-03 10:01:43 +00:00
log . msg ( " download finished " )
self . failUnlessEqual ( data , DATA )
2007-04-19 01:29:10 +00:00
d . addCallback ( _download_to_data_done )
2007-04-16 20:07:36 +00:00
target_filename = os . path . join ( self . basedir , " download.target " )
def _download_to_filename ( res ) :
return self . downloader . download_to_filename ( self . uri ,
target_filename )
d . addCallback ( _download_to_filename )
def _download_to_filename_done ( res ) :
newdata = open ( target_filename , " rb " ) . read ( )
self . failUnlessEqual ( newdata , DATA )
d . addCallback ( _download_to_filename_done )
target_filename2 = os . path . join ( self . basedir , " download.target2 " )
def _download_to_filehandle ( res ) :
fh = open ( target_filename2 , " wb " )
return self . downloader . download_to_filehandle ( self . uri , fh )
d . addCallback ( _download_to_filehandle )
def _download_to_filehandle_done ( fh ) :
fh . close ( )
newdata = open ( target_filename2 , " rb " ) . read ( )
self . failUnlessEqual ( newdata , DATA )
d . addCallback ( _download_to_filehandle_done )
2007-04-19 01:29:10 +00:00
def _download_nonexistent_uri ( res ) :
baduri = self . mangle_uri ( self . uri )
d1 = self . downloader . download_to_data ( baduri )
def _baduri_should_fail ( res ) :
self . failUnless ( isinstance ( res , Failure ) )
2007-04-26 00:53:10 +00:00
self . failUnless ( res . check ( download . NotEnoughPeersError ) ,
" expected NotEnoughPeersError, got %s " % res )
2007-04-19 01:29:10 +00:00
# TODO: files that have zero peers should get a special kind
# of NotEnoughPeersError, which can be used to suggest that
# the URI might be wrong or that they've nver uploaded the
# file in the first place.
d1 . addBoth ( _baduri_should_fail )
return d1
d . addCallback ( _download_nonexistent_uri )
2006-12-03 02:37:31 +00:00
return d
2007-05-02 22:31:05 +00:00
test_upload_and_download . timeout = 4800
2006-12-03 07:56:27 +00:00
2007-04-19 01:29:10 +00:00
def flip_bit ( self , good ) :
return good [ : - 1 ] + chr ( ord ( good [ - 1 ] ) ^ 0x01 )
def mangle_uri ( self , gooduri ) :
2007-07-22 01:23:15 +00:00
# change the key, which changes the storage index, which means we'll
# be asking about the wrong file, so nobody will have any shares
2007-07-21 22:40:36 +00:00
u = IFileURI ( gooduri )
2007-07-22 01:23:15 +00:00
u2 = uri . CHKFileURI ( key = self . flip_bit ( u . key ) ,
2007-07-21 22:40:36 +00:00
uri_extension_hash = u . uri_extension_hash ,
needed_shares = u . needed_shares ,
total_shares = u . total_shares ,
size = u . size )
return u2 . to_string ( )
2007-04-19 01:29:10 +00:00
2007-06-08 22:59:16 +00:00
# TODO: add a test which mangles the uri_extension_hash instead, and
# should fail due to not being able to get a valid uri_extension block.
# Also a test which sneakily mangles the uri_extension block to change
# some of the validation data, so it will fail in the post-download phase
# when the file's crypttext integrity check fails. Do the same thing for
# the key, which should cause the download to fail the post-download
2007-06-10 03:46:04 +00:00
# plaintext_hash check.
2007-04-26 00:53:10 +00:00
2006-12-04 05:42:19 +00:00
def test_vdrive ( self ) :
2007-06-28 18:00:03 +00:00
self . basedir = " system/SystemTest/test_vdrive "
2007-07-12 20:22:36 +00:00
self . data = LARGE_DATA
2006-12-04 05:42:19 +00:00
d = self . set_up_nodes ( )
2007-06-28 18:00:03 +00:00
d . addCallback ( self . log , " starting publish " )
d . addCallback ( self . _do_publish )
# at this point, we have the following global filesystem:
# /
# /subdir1
# /subdir1/mydata567
2007-06-28 22:05:33 +00:00
# /subdir1/subdir2/
# /subdir1/subdir2/mydata992
2007-06-28 18:00:03 +00:00
d . addCallback ( self . _bounce_client0 )
d . addCallback ( self . log , " bounced client0 " )
d . addCallback ( self . _check_publish1 )
d . addCallback ( self . log , " did _check_publish1 " )
d . addCallback ( self . _check_publish2 )
d . addCallback ( self . log , " did _check_publish2 " )
d . addCallback ( self . _do_publish_private )
d . addCallback ( self . log , " did _do_publish_private " )
# now we also have:
# ~client0/personal/sekrit data
2007-06-28 22:05:33 +00:00
# ~client0/s2-rw -> /subdir1/subdir2/
# ~client0/s2-ro -> /subdir1/subdir2/ (read-only)
2007-06-28 18:00:03 +00:00
d . addCallback ( self . _check_publish_private )
d . addCallback ( self . log , " did _check_publish_private " )
2006-12-07 19:48:06 +00:00
d . addCallback ( self . _test_web )
2007-06-26 22:55:00 +00:00
d . addCallback ( self . _test_runner )
2006-12-04 05:42:19 +00:00
return d
2007-05-01 03:14:58 +00:00
test_vdrive . timeout = 1100
2006-12-04 05:42:19 +00:00
2007-06-28 18:00:03 +00:00
def _do_publish ( self , res ) :
ut = upload . Data ( self . data )
c0 = self . clients [ 0 ]
d = c0 . getServiceNamed ( " vdrive " ) . get_public_root ( )
d . addCallback ( lambda root : root . create_empty_directory ( " subdir1 " ) )
2007-06-28 22:05:33 +00:00
def _made_subdir1 ( subdir1_node ) :
d1 = subdir1_node . add_file ( " mydata567 " , ut )
d1 . addCallback ( self . log , " publish finished " )
def _stash_uri ( filenode ) :
self . uri = filenode . get_uri ( )
d1 . addCallback ( _stash_uri )
d1 . addCallback ( lambda res :
subdir1_node . create_empty_directory ( " subdir2 " ) )
d1 . addCallback ( lambda subdir2 :
subdir2 . add_file ( " mydata992 " , ut ) )
return d1
d . addCallback ( _made_subdir1 )
2007-06-28 18:00:03 +00:00
return d
def _bounce_client0 ( self , res ) :
old_client0 = self . clients [ 0 ]
d = old_client0 . disownServiceParent ( )
assert isinstance ( d , defer . Deferred )
d . addCallback ( self . log , " STOPPED " )
2007-06-29 02:20:28 +00:00
# I think windows requires a moment to let the connection really stop
# and the port number made available for re-use. TODO: examine the
# behavior, see if this is really the problem, see if we can do
# better than blindly waiting for a second.
d . addCallback ( self . stall , 1.0 )
2007-06-28 18:00:03 +00:00
def _stopped ( res ) :
new_client0 = client . Client ( basedir = self . getdir ( " client0 " ) )
self . add_service ( new_client0 )
self . clients [ 0 ] = new_client0
return self . wait_for_connections ( )
d . addCallback ( _stopped )
d . addCallback ( self . log , " CONNECTED " )
def _connected ( res ) :
# now find out where the web port was
l = self . clients [ 0 ] . getServiceNamed ( " webish " ) . listener
port = l . _port . getHost ( ) . port
self . webish_url = " http://localhost: %d / " % port
d . addCallback ( _connected )
d . addCallback ( self . log , " GOT WEB LISTENER " )
return d
def log ( self , res , msg ) :
2007-06-28 22:05:33 +00:00
#print "MSG: %s RES: %s" % (msg, res)
2007-06-28 18:00:03 +00:00
log . msg ( msg )
return res
2007-06-29 02:20:28 +00:00
def stall ( self , res , delay = 1.0 ) :
d = defer . Deferred ( )
reactor . callLater ( delay , d . callback , res )
return d
2007-06-28 18:00:03 +00:00
def _do_publish_private ( self , res ) :
2007-07-12 23:17:49 +00:00
self . smalldata = " sssh, very secret stuff "
ut = upload . Data ( self . smalldata )
2007-06-28 18:00:03 +00:00
vdrive0 = self . clients [ 0 ] . getServiceNamed ( " vdrive " )
2007-06-29 00:46:14 +00:00
d = vdrive0 . get_node_at_path ( " ~ " )
2007-06-28 18:00:03 +00:00
d . addCallback ( self . log , " GOT ~ " )
2007-06-28 22:05:33 +00:00
def _got_root ( rootnode ) :
d1 = rootnode . create_empty_directory ( " personal " )
d1 . addCallback ( self . log , " made ~/personal " )
d1 . addCallback ( lambda node : node . add_file ( " sekrit data " , ut ) )
d1 . addCallback ( self . log , " made ~/personal/sekrit data " )
d1 . addCallback ( lambda res :
vdrive0 . get_node_at_path ( [ " subdir1 " , " subdir2 " ] ) )
def _got_s2 ( s2node ) :
d2 = rootnode . set_uri ( " s2-rw " , s2node . get_uri ( ) )
d2 . addCallback ( lambda node :
rootnode . set_uri ( " s2-ro " ,
s2node . get_immutable_uri ( ) ) )
return d2
d1 . addCallback ( _got_s2 )
return d1
d . addCallback ( _got_root )
2007-06-28 18:00:03 +00:00
return d
def _check_publish1 ( self , res ) :
# this one uses the iterative API
c1 = self . clients [ 1 ]
d = c1 . getServiceNamed ( " vdrive " ) . get_public_root ( )
d . addCallback ( self . log , " check_publish1 got / " )
d . addCallback ( lambda root : root . get ( " subdir1 " ) )
d . addCallback ( lambda subdir1 : subdir1 . get ( " mydata567 " ) )
d . addCallback ( lambda filenode : filenode . download_to_data ( ) )
d . addCallback ( self . log , " get finished " )
def _get_done ( data ) :
self . failUnlessEqual ( data , self . data )
d . addCallback ( _get_done )
return d
def _check_publish2 ( self , res ) :
# this one uses the path-based API
vdrive1 = self . clients [ 1 ] . getServiceNamed ( " vdrive " )
2007-06-29 00:46:14 +00:00
get_path = vdrive1 . get_node_at_path
2007-06-28 18:00:03 +00:00
d = get_path ( " subdir1 " )
d . addCallback ( lambda dirnode :
self . failUnless ( IDirectoryNode . providedBy ( dirnode ) ) )
2007-06-29 00:46:14 +00:00
d . addCallback ( lambda res : get_path ( " /subdir1/mydata567 " ) )
2007-06-28 18:00:03 +00:00
d . addCallback ( lambda filenode : filenode . download_to_data ( ) )
d . addCallback ( lambda data : self . failUnlessEqual ( data , self . data ) )
d . addCallback ( lambda res : get_path ( " subdir1/mydata567 " ) )
def _got_filenode ( filenode ) :
d1 = vdrive1 . get_node ( filenode . get_uri ( ) )
d1 . addCallback ( self . failUnlessEqual , filenode )
return d1
d . addCallback ( _got_filenode )
return d
def _check_publish_private ( self , res ) :
# this one uses the path-based API
def get_path ( path ) :
vdrive0 = self . clients [ 0 ] . getServiceNamed ( " vdrive " )
2007-06-29 00:46:14 +00:00
return vdrive0 . get_node_at_path ( path )
2007-06-28 18:00:03 +00:00
d = get_path ( " ~/personal " )
2007-06-28 22:05:33 +00:00
def _got_personal ( personal ) :
self . _personal_node = personal
return personal
d . addCallback ( _got_personal )
2007-06-28 18:00:03 +00:00
d . addCallback ( lambda dirnode :
self . failUnless ( IDirectoryNode . providedBy ( dirnode ) ) )
d . addCallback ( lambda res : get_path ( " ~/personal/sekrit data " ) )
d . addCallback ( lambda filenode : filenode . download_to_data ( ) )
2007-07-12 23:17:49 +00:00
d . addCallback ( lambda data : self . failUnlessEqual ( data , self . smalldata ) )
2007-06-28 22:05:33 +00:00
d . addCallback ( lambda res : get_path ( " ~/s2-rw " ) )
d . addCallback ( lambda dirnode : self . failUnless ( dirnode . is_mutable ( ) ) )
d . addCallback ( lambda res : get_path ( " ~/s2-ro " ) )
def _got_s2ro ( dirnode ) :
self . failIf ( dirnode . is_mutable ( ) )
d1 = defer . succeed ( None )
d1 . addCallback ( lambda res : dirnode . list ( ) )
d1 . addCallback ( self . log , " dirnode.list " )
d1 . addCallback ( lambda res : dirnode . create_empty_directory ( " nope " ) )
d1 . addBoth ( self . shouldFail , NotMutableError , " mkdir(nope) " )
d1 . addCallback ( self . log , " doing add_file(ro) " )
ut = upload . Data ( " I will disappear, unrecorded and unobserved. The tragedy of my demise is made more poignant by its silence, but this beauty is not for you to ever know. " )
d1 . addCallback ( lambda res : dirnode . add_file ( " hope " , ut ) )
d1 . addBoth ( self . shouldFail , NotMutableError , " add_file(nope) " )
d1 . addCallback ( self . log , " doing get(ro) " )
d1 . addCallback ( lambda res : dirnode . get ( " mydata992 " ) )
d1 . addCallback ( lambda filenode :
self . failUnless ( IFileNode . providedBy ( filenode ) ) )
d1 . addCallback ( self . log , " doing delete(ro) " )
d1 . addCallback ( lambda res : dirnode . delete ( " mydata992 " ) )
d1 . addBoth ( self . shouldFail , NotMutableError , " delete(nope) " )
d1 . addCallback ( lambda res : dirnode . set_uri ( " hopeless " , self . uri ) )
d1 . addBoth ( self . shouldFail , NotMutableError , " set_uri(nope) " )
d1 . addCallback ( lambda res : dirnode . get ( " missing " ) )
2007-07-07 02:43:03 +00:00
d1 . addBoth ( self . shouldFail , KeyError , " get(missing) " ,
2007-06-28 22:05:33 +00:00
" unable to find child named ' missing ' " )
d1 . addCallback ( self . log , " doing move_child_to(ro) " )
personal = self . _personal_node
d1 . addCallback ( lambda res :
dirnode . move_child_to ( " mydata992 " ,
personal , " nope " ) )
d1 . addBoth ( self . shouldFail , NotMutableError , " mv from readonly " )
d1 . addCallback ( self . log , " doing move_child_to(ro)2 " )
d1 . addCallback ( lambda res :
personal . move_child_to ( " sekrit data " ,
dirnode , " nope " ) )
d1 . addBoth ( self . shouldFail , NotMutableError , " mv to readonly " )
d1 . addCallback ( self . log , " finished with _got_s2ro " )
return d1
d . addCallback ( _got_s2ro )
d . addCallback ( lambda res : get_path ( " ~ " ) )
def _got_home ( home ) :
personal = self . _personal_node
d1 = defer . succeed ( None )
d1 . addCallback ( self . log , " mv ' ~/personal/sekrit data ' to ~/sekrit " )
d1 . addCallback ( lambda res :
personal . move_child_to ( " sekrit data " , home , " sekrit " ) )
d1 . addCallback ( self . log , " mv ~/sekrit ' ~/sekrit data ' " )
d1 . addCallback ( lambda res :
home . move_child_to ( " sekrit " , home , " sekrit data " ) )
d1 . addCallback ( self . log , " mv ' ~/sekret data ' ~/personal/ " )
d1 . addCallback ( lambda res :
home . move_child_to ( " sekrit data " , personal ) )
d1 . addCallback ( lambda res : home . build_manifest ( ) )
d1 . addCallback ( self . log , " manifest " )
# four items:
# ~client0/personal/
# ~client0/personal/sekrit data
# ~client0/s2-rw (same as ~client/s2-ro)
# ~client0/s2-rw/mydata992 (same as ~client/s2-rw/mydata992)
d1 . addCallback ( lambda manifest :
self . failUnlessEqual ( len ( manifest ) , 4 ) )
return d1
d . addCallback ( _got_home )
2007-06-28 18:00:03 +00:00
return d
2007-06-28 22:05:33 +00:00
def shouldFail ( self , res , expected_failure , which , substring = None ) :
if isinstance ( res , Failure ) :
res . trap ( expected_failure )
if substring :
self . failUnless ( substring in str ( res ) ,
" substring ' %s ' not in ' %s ' "
% ( substring , str ( res ) ) )
else :
self . fail ( " %s was supposed to raise %s , not get ' %s ' " %
( which , expected_failure , res ) )
2006-12-07 19:48:06 +00:00
def _test_web ( self , res ) :
base = self . webish_url
d = getPage ( base )
def _got_welcome ( page ) :
2007-01-16 22:12:49 +00:00
expected = " Connected Peers: <span> %d </span> " % ( self . numclients )
2006-12-07 19:48:06 +00:00
self . failUnless ( expected in page ,
" I didn ' t see the right ' connected peers ' message "
" in: %s " % page
)
2007-03-29 21:31:55 +00:00
expected = " My nodeid: <span> %s </span> " % idlib . b2a ( self . clients [ 0 ] . nodeid )
self . failUnless ( expected in page ,
" I didn ' t see the right ' My nodeid ' message "
" in: %s " % page )
2006-12-07 19:48:06 +00:00
d . addCallback ( _got_welcome )
2007-06-15 08:38:55 +00:00
d . addCallback ( lambda res : getPage ( base + " global_vdrive " ) )
d . addCallback ( lambda res : getPage ( base + " global_vdrive/subdir1 " ) )
2006-12-07 19:48:06 +00:00
def _got_subdir1 ( page ) :
# there ought to be an href for our file
2007-05-24 00:25:49 +00:00
self . failUnless ( ( " <td> %d </td> " % len ( self . data ) ) in page )
2006-12-07 21:48:37 +00:00
self . failUnless ( " >mydata567</a> " in page )
2006-12-07 19:48:06 +00:00
d . addCallback ( _got_subdir1 )
2007-06-15 08:38:55 +00:00
d . addCallback ( lambda res :
getPage ( base + " global_vdrive/subdir1/mydata567 " ) )
2006-12-07 21:48:37 +00:00
def _got_data ( page ) :
self . failUnlessEqual ( page , self . data )
d . addCallback ( _got_data )
2007-04-24 08:41:54 +00:00
# download from a URI embedded in a URL
def _get_from_uri ( res ) :
return getPage ( base + " download_uri/ %s ?filename= %s "
% ( self . uri , " mydata567 " ) )
d . addCallback ( _get_from_uri )
def _got_from_uri ( page ) :
self . failUnlessEqual ( page , self . data )
d . addCallback ( _got_from_uri )
# download from a URI embedded in a URL, second form
def _get_from_uri2 ( res ) :
return getPage ( base + " download_uri?uri= %s " % ( self . uri , ) )
d . addCallback ( _get_from_uri2 )
def _got_from_uri2 ( page ) :
self . failUnlessEqual ( page , self . data )
d . addCallback ( _got_from_uri2 )
2007-07-03 20:18:14 +00:00
# download from a bogus URI, make sure we get a reasonable error
def _get_from_bogus_uri ( res ) :
return getPage ( base + " download_uri/ %s ?filename= %s "
% ( self . mangle_uri ( self . uri ) , " mydata567 " ) )
d . addCallback ( _get_from_bogus_uri )
2007-07-03 20:47:37 +00:00
d . addBoth ( self . shouldFail , Error , " downloading bogus URI " ,
2007-07-08 03:06:44 +00:00
" 404 Not Found " )
2007-07-03 20:47:37 +00:00
# TODO: mangle the second segment of a file, to test errors that
# occur after we've already sent some good data, which uses a
# different error path.
2007-07-03 20:18:14 +00:00
2007-07-08 03:06:44 +00:00
# TODO: download a URI with a form
2007-04-24 08:41:54 +00:00
# TODO: create a directory by using a form
# TODO: upload by using a form on the directory page
2007-06-15 08:38:55 +00:00
# url = base + "global_vdrive/subdir1/freeform_post!!upload"
2007-04-24 08:41:54 +00:00
# TODO: delete a file by using a button on the directory page
2006-12-07 19:48:06 +00:00
return d
2007-06-26 22:55:00 +00:00
def _test_runner ( self , res ) :
# exercise some of the diagnostic tools in runner.py
2007-07-13 23:58:08 +00:00
# find a share
2007-06-26 22:55:00 +00:00
for ( dirpath , dirnames , filenames ) in os . walk ( self . basedir ) :
2007-07-13 23:58:08 +00:00
if " storage " not in dirpath :
continue
if not filenames :
continue
pieces = dirpath . split ( os . sep )
if pieces [ - 2 ] == " storage " :
# we're sitting in .../storage/$SINDEX , and there are
# sharefiles here
filename = os . path . join ( dirpath , filenames [ 0 ] )
2007-06-26 22:55:00 +00:00
break
else :
self . fail ( " unable to find any uri_extension files in %s "
% self . basedir )
2007-07-13 23:58:08 +00:00
log . msg ( " test_system.SystemTest._test_runner using %s " % filename )
2007-06-26 23:51:38 +00:00
out , err = StringIO ( ) , StringIO ( )
rc = runner . runner ( [ " dump-uri-extension " ,
filename ] ,
stdout = out , stderr = err )
output = out . getvalue ( )
2007-06-26 22:55:00 +00:00
self . failUnlessEqual ( rc , 0 )
# we only upload a single file, so we can assert some things about
# its size and shares
self . failUnless ( " size: %d \n " % len ( self . data ) in output )
self . failUnless ( " num_segments: 1 \n " in output )
# segment_size is always a multiple of needed_shares
2007-07-12 22:33:30 +00:00
self . failUnless ( " segment_size: 114 \n " in output )
self . failUnless ( " total_shares: 10 \n " in output )
2007-06-26 22:55:00 +00:00
# keys which are supposed to be present
for key in ( " size " , " num_segments " , " segment_size " ,
" needed_shares " , " total_shares " ,
" codec_name " , " codec_params " , " tail_codec_params " ,
" plaintext_hash " , " plaintext_root_hash " ,
" crypttext_hash " , " crypttext_root_hash " ,
" share_root_hash " ,
) :
self . failUnless ( " %s : " % key in output , key )