2006-12-03 01:30:18 +00:00
2007-08-12 23:33:51 +00:00
from base64 import b32encode
2007-11-08 00:51:35 +00:00
import os , sys , time , re
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
2007-10-12 02:20:41 +00:00
from twisted . internet import threads # CLI tests use deferToThread
2006-12-03 01:30:18 +00:00
from twisted . application import service
2007-11-14 06:08:15 +00:00
from allmydata import client , uri , download , upload , storage , mutable
2007-12-03 21:52:42 +00:00
from allmydata . introducer import IntroducerNode
from allmydata . util import deferredutil , fileutil , idlib , mathutil , 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-12-03 21:52:42 +00:00
from allmydata . mutable 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
2007-12-03 21:52:42 +00:00
def set_up_nodes ( self , NUMCLIENTS = 5 , createprivdir = False ) :
2006-12-03 07:56:27 +00:00
self . numclients = NUMCLIENTS
2007-12-03 21:52:42 +00:00
self . createprivdir = createprivdir
iv_dir = self . getdir ( " introducer " )
2007-06-01 01:31:01 +00:00
if not os . path . isdir ( iv_dir ) :
fileutil . make_dirs ( iv_dir )
2007-12-03 21:52:42 +00:00
iv = IntroducerNode ( basedir = iv_dir )
self . introducer = self . add_service ( iv )
d = self . introducer . 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-12-03 21:52:42 +00:00
q = self . introducer
self . introducer_furl = q . introducer_url
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 )
2007-12-03 21:52:42 +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-12-03 21:52:42 +00:00
if self . createprivdir :
open ( os . path . join ( basedir , " my_private_dir.uri " ) , " w " )
2007-04-30 16:57:52 +00:00
open ( os . path . join ( basedir , " introducer.furl " ) , " w " ) . write ( self . introducer_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
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 ) :
2007-10-15 23:16:39 +00:00
# TODO: replace this with something that takes a list of peerids and
# fires when they've all been heard from, instead of using a count
# and a threshold
2006-12-03 07:56:27 +00:00
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-12-03 21:52:42 +00:00
d . addCallback ( lambda res : self . add_extra_node ( self . numclients ) )
2007-01-10 02:40:36 +00:00
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-08-12 23:29:34 +00:00
all_peerids = list ( c . get_all_peerids ( ) )
2007-12-03 21:52:42 +00:00
self . failUnlessEqual ( len ( all_peerids ) , self . numclients + 1 )
2007-08-12 23:29:34 +00:00
permuted_peers = list ( c . get_permuted_peers ( " a " , True ) )
2007-12-03 21:52:42 +00:00
self . failUnlessEqual ( len ( permuted_peers ) , self . numclients + 1 )
2007-08-12 23:29:34 +00:00
permuted_other_peers = list ( c . get_permuted_peers ( " a " , False ) )
2007-12-03 21:52:42 +00:00
self . failUnlessEqual ( len ( permuted_other_peers ) , self . numclients )
2007-08-12 23:29:34 +00:00
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 ( )
2007-08-12 23:29:34 +00:00
def _check_connections ( res ) :
for c in self . clients :
all_peerids = list ( c . get_all_peerids ( ) )
2007-12-03 21:52:42 +00:00
self . failUnlessEqual ( len ( all_peerids ) , self . numclients )
2007-08-12 23:29:34 +00:00
permuted_peers = list ( c . get_permuted_peers ( " a " , True ) )
2007-12-03 21:52:42 +00:00
self . failUnlessEqual ( len ( permuted_peers ) , self . numclients )
2007-08-12 23:29:34 +00:00
permuted_other_peers = list ( c . get_permuted_peers ( " a " , False ) )
2007-12-03 21:52:42 +00:00
self . failUnlessEqual ( len ( permuted_other_peers ) , self . numclients - 1 )
2007-08-12 23:29:34 +00:00
d . addCallback ( _check_connections )
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
2007-09-26 19:07:37 +00:00
# the URI might be wrong or that they've never uploaded the
2007-04-19 01:29:10 +00:00
# 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-11-14 06:08:15 +00:00
def _find_shares ( self , basedir ) :
shares = [ ]
for ( dirpath , dirnames , filenames ) in os . walk ( basedir ) :
if " storage " not in dirpath :
continue
if not filenames :
continue
pieces = dirpath . split ( os . sep )
if pieces [ - 3 ] == " storage " and pieces [ - 2 ] == " shares " :
# we're sitting in .../storage/shares/$SINDEX , and there
# are sharefiles here
assert pieces [ - 4 ] . startswith ( " client " )
client_num = int ( pieces [ - 4 ] [ - 1 ] )
storage_index_s = pieces [ - 1 ]
storage_index = idlib . a2b ( storage_index_s )
for sharename in filenames :
shnum = int ( sharename )
filename = os . path . join ( dirpath , sharename )
data = ( client_num , storage_index , filename , shnum )
shares . append ( data )
if not shares :
self . fail ( " unable to find any share files in %s " % basedir )
return shares
def _corrupt_mutable_share ( self , filename , which ) :
msf = storage . MutableShareFile ( filename )
datav = msf . readv ( [ ( 0 , 1000000 ) ] )
final_share = datav [ 0 ]
assert len ( final_share ) < 1000000 # ought to be truncated
pieces = mutable . unpack_share ( final_share )
( seqnum , root_hash , IV , k , N , segsize , datalen ,
verification_key , signature , share_hash_chain , block_hash_tree ,
share_data , enc_privkey ) = pieces
if which == " seqnum " :
seqnum = seqnum + 15
elif which == " R " :
root_hash = self . flip_bit ( root_hash )
elif which == " IV " :
IV = self . flip_bit ( IV )
elif which == " segsize " :
segsize = segsize + 15
elif which == " pubkey " :
verification_key = self . flip_bit ( verification_key )
elif which == " signature " :
signature = self . flip_bit ( signature )
elif which == " share_hash_chain " :
nodenum = share_hash_chain . keys ( ) [ 0 ]
share_hash_chain [ nodenum ] = self . flip_bit ( share_hash_chain [ nodenum ] )
elif which == " block_hash_tree " :
block_hash_tree [ - 1 ] = self . flip_bit ( block_hash_tree [ - 1 ] )
elif which == " share_data " :
share_data = self . flip_bit ( share_data )
elif which == " encprivkey " :
enc_privkey = self . flip_bit ( enc_privkey )
prefix = mutable . pack_prefix ( seqnum , root_hash , IV , k , N ,
segsize , datalen )
final_share = mutable . pack_share ( prefix ,
verification_key ,
signature ,
share_hash_chain ,
block_hash_tree ,
share_data ,
enc_privkey )
msf . writev ( [ ( 0 , final_share ) ] , None )
2007-11-07 01:57:11 +00:00
def test_mutable ( self ) :
self . basedir = " system/SystemTest/test_mutable "
2007-11-07 21:19:01 +00:00
DATA = " initial contents go here. " # 25 bytes % 3 != 0
2007-11-08 04:01:39 +00:00
NEWDATA = " new contents yay "
NEWERDATA = " this is getting old "
2007-11-07 01:57:11 +00:00
d = self . set_up_nodes ( )
2007-11-15 21:55:00 +00:00
2007-11-07 01:57:11 +00:00
def _create_mutable ( res ) :
c = self . clients [ 0 ]
2007-11-08 11:07:33 +00:00
log . msg ( " starting create_mutable_file " )
2007-12-03 21:52:42 +00:00
d1 = c . create_mutable_file ( DATA , wait_for_numpeers = self . numclients )
2007-11-07 01:57:11 +00:00
def _done ( res ) :
log . msg ( " DONE: %s " % ( res , ) )
2007-11-07 21:19:01 +00:00
self . _mutable_node_1 = res
uri = res . get_uri ( )
2007-11-08 04:01:39 +00:00
d1 . addCallback ( _done )
2007-11-07 01:57:11 +00:00
return d1
d . addCallback ( _create_mutable )
2007-11-07 02:10:49 +00:00
def _test_debug ( res ) :
# find a share. It is important to run this while there is only
# one slot in the grid.
2007-11-14 06:08:15 +00:00
shares = self . _find_shares ( self . basedir )
( client_num , storage_index , filename , shnum ) = shares [ 0 ]
2007-11-07 02:10:49 +00:00
log . msg ( " test_system.SystemTest.test_mutable._test_debug using %s "
% filename )
log . msg ( " for clients[ %d ] " % client_num )
out , err = StringIO ( ) , StringIO ( )
rc = runner . runner ( [ " dump-share " ,
filename ] ,
stdout = out , stderr = err )
output = out . getvalue ( )
self . failUnlessEqual ( rc , 0 )
2007-11-07 02:50:33 +00:00
try :
self . failUnless ( " Mutable slot found: \n " in output )
self . failUnless ( " share_type: SDMF \n " in output )
peerid = idlib . nodeid_b2a ( self . clients [ client_num ] . nodeid )
self . failUnless ( " WE for nodeid: %s \n " % peerid in output )
self . failUnless ( " num_extra_leases: 0 \n " in output )
2007-11-08 00:51:35 +00:00
# the pubkey size can vary by a byte, so the container might
# be a bit larger on some runs.
m = re . search ( r ' ^ container_size: ( \ d+)$ ' , output , re . M )
self . failUnless ( m )
container_size = int ( m . group ( 1 ) )
2007-12-03 21:52:42 +00:00
self . failUnless ( 2037 < = container_size < = 2049 , container_size )
2007-11-08 00:51:35 +00:00
m = re . search ( r ' ^ data_length: ( \ d+)$ ' , output , re . M )
self . failUnless ( m )
data_length = int ( m . group ( 1 ) )
2007-12-03 21:52:42 +00:00
self . failUnless ( 2037 < = data_length < = 2049 , data_length )
2007-11-07 02:50:33 +00:00
self . failUnless ( " secrets are for nodeid: %s \n " % peerid
in output )
self . failUnless ( " SDMF contents: \n " in output )
self . failUnless ( " seqnum: 1 \n " in output )
self . failUnless ( " required_shares: 3 \n " in output )
self . failUnless ( " total_shares: 10 \n " in output )
2007-12-03 21:52:42 +00:00
self . failUnless ( " segsize: 27 \n " in output , ( output , filename ) )
2007-11-07 02:50:33 +00:00
self . failUnless ( " datalen: 25 \n " in output )
# the exact share_hash_chain nodes depends upon the sharenum,
# and is more of a hassle to compute than I want to deal with
# now
self . failUnless ( " share_hash_chain: " in output )
self . failUnless ( " block_hash_tree: 1 nodes \n " in output )
except unittest . FailTest :
print
print " dump-share output was: "
print output
raise
2007-11-07 02:10:49 +00:00
d . addCallback ( _test_debug )
2007-11-07 21:19:01 +00:00
# test retrieval
# first, let's see if we can use the existing node to retrieve the
# contents. This allows it to use the cached pubkey and maybe the
# latest-known sharemap.
d . addCallback ( lambda res : self . _mutable_node_1 . download_to_data ( ) )
def _check_download_1 ( res ) :
self . failUnlessEqual ( res , DATA )
# now we see if we can retrieve the data from a new node,
# constructed using the URI of the original one. We do this test
# on the same client that uploaded the data.
uri = self . _mutable_node_1 . get_uri ( )
2007-11-08 11:07:33 +00:00
log . msg ( " starting retrieve1 " )
2007-11-09 09:54:51 +00:00
newnode = self . clients [ 0 ] . create_node_from_uri ( uri )
2007-11-07 21:19:01 +00:00
return newnode . download_to_data ( )
d . addCallback ( _check_download_1 )
def _check_download_2 ( res ) :
self . failUnlessEqual ( res , DATA )
# same thing, but with a different client
uri = self . _mutable_node_1 . get_uri ( )
2007-11-09 09:54:51 +00:00
newnode = self . clients [ 1 ] . create_node_from_uri ( uri )
2007-11-08 11:07:33 +00:00
log . msg ( " starting retrieve2 " )
2007-11-08 04:01:39 +00:00
d1 = newnode . download_to_data ( )
d1 . addCallback ( lambda res : ( res , newnode ) )
return d1
2007-11-07 21:19:01 +00:00
d . addCallback ( _check_download_2 )
2007-11-08 04:01:39 +00:00
def _check_download_3 ( ( res , newnode ) ) :
2007-11-07 21:19:01 +00:00
self . failUnlessEqual ( res , DATA )
2007-11-08 04:01:39 +00:00
# replace the data
2007-11-08 11:07:33 +00:00
log . msg ( " starting replace1 " )
2007-12-03 21:52:42 +00:00
d1 = newnode . replace ( NEWDATA , wait_for_numpeers = self . numclients )
2007-11-08 04:01:39 +00:00
d1 . addCallback ( lambda res : newnode . download_to_data ( ) )
return d1
2007-11-07 21:19:01 +00:00
d . addCallback ( _check_download_3 )
2007-11-08 04:01:39 +00:00
def _check_download_4 ( res ) :
self . failUnlessEqual ( res , NEWDATA )
# now create an even newer node and replace the data on it. This
# new node has never been used for download before.
uri = self . _mutable_node_1 . get_uri ( )
2007-11-09 09:54:51 +00:00
newnode1 = self . clients [ 2 ] . create_node_from_uri ( uri )
newnode2 = self . clients [ 3 ] . create_node_from_uri ( uri )
2007-11-14 06:08:15 +00:00
self . _newnode3 = self . clients [ 3 ] . create_node_from_uri ( uri )
2007-11-08 11:07:33 +00:00
log . msg ( " starting replace2 " )
2007-12-03 21:52:42 +00:00
d1 = newnode1 . replace ( NEWERDATA , wait_for_numpeers = self . numclients )
2007-11-08 04:01:39 +00:00
d1 . addCallback ( lambda res : newnode2 . download_to_data ( ) )
return d1
2007-11-08 11:07:33 +00:00
d . addCallback ( _check_download_4 )
2007-11-08 04:01:39 +00:00
def _check_download_5 ( res ) :
2007-11-08 11:07:33 +00:00
log . msg ( " finished replace2 " )
2007-11-08 04:01:39 +00:00
self . failUnlessEqual ( res , NEWERDATA )
2007-12-03 22:21:14 +00:00
d . addCallback ( _check_download_5 )
def _corrupt_shares ( res ) :
# run around and flip bits in all but k of the shares, to test
# the hash checks
shares = self . _find_shares ( self . basedir )
## sort by share number
#shares.sort( lambda a,b: cmp(a[3], b[3]) )
where = dict ( [ ( shnum , filename )
for ( client_num , storage_index , filename , shnum )
in shares ] )
assert len ( where ) == 10 # this test is designed for 3-of-10
for shnum , filename in where . items ( ) :
# shares 7,8,9 are left alone. read will check
# (share_hash_chain, block_hash_tree, share_data). New
# seqnum+R pairs will trigger a check of (seqnum, R, IV,
# segsize, signature).
if shnum == 0 :
# read: this will trigger "pubkey doesn't match
# fingerprint".
self . _corrupt_mutable_share ( filename , " pubkey " )
self . _corrupt_mutable_share ( filename , " encprivkey " )
elif shnum == 1 :
# triggers "signature is invalid"
self . _corrupt_mutable_share ( filename , " seqnum " )
elif shnum == 2 :
# triggers "signature is invalid"
self . _corrupt_mutable_share ( filename , " R " )
elif shnum == 3 :
# triggers "signature is invalid"
self . _corrupt_mutable_share ( filename , " segsize " )
elif shnum == 4 :
self . _corrupt_mutable_share ( filename , " share_hash_chain " )
elif shnum == 5 :
self . _corrupt_mutable_share ( filename , " block_hash_tree " )
elif shnum == 6 :
self . _corrupt_mutable_share ( filename , " share_data " )
# other things to correct: IV, signature
# 7,8,9 are left alone
# note that initial_query_count=5 means that we'll hit the
# first 5 servers in effectively random order (based upon
# response time), so we won't necessarily ever get a "pubkey
# doesn't match fingerprint" error (if we hit shnum>=1 before
# shnum=0, we pull the pubkey from there). To get repeatable
# specific failures, we need to set initial_query_count=1,
# but of course that will change the sequencing behavior of
# the retrieval process. TODO: find a reasonable way to make
# this a parameter, probably when we expand this test to test
# for one failure mode at a time.
# when we retrieve this, we should get three signature
# failures (where we've mangled seqnum, R, and segsize). The
# pubkey mangling
d . addCallback ( _corrupt_shares )
d . addCallback ( lambda res : self . _newnode3 . download_to_data ( ) )
d . addCallback ( _check_download_5 )
def _check_empty_file ( res ) :
2007-11-14 06:08:15 +00:00
# make sure we can create empty files, this usually screws up the
# segsize math
2007-12-03 22:21:14 +00:00
d1 = self . clients [ 2 ] . create_mutable_file ( " " , wait_for_numpeers = self . numclients )
2007-11-08 11:31:00 +00:00
d1 . addCallback ( lambda newnode : newnode . download_to_data ( ) )
d1 . addCallback ( lambda res : self . failUnlessEqual ( " " , res ) )
return d1
2007-11-14 06:08:15 +00:00
d . addCallback ( _check_empty_file )
2007-11-08 04:01:39 +00:00
2007-12-03 21:52:42 +00:00
d . addCallback ( lambda res : self . clients [ 0 ] . create_empty_dirnode ( wait_for_numpeers = self . numclients ) )
2007-11-08 11:31:00 +00:00
def _created_dirnode ( dnode ) :
2007-12-03 21:52:42 +00:00
log . msg ( " _created_dirnode( %s ) " % ( dnode , ) )
2007-11-08 11:31:00 +00:00
d1 = dnode . list ( )
d1 . addCallback ( lambda children : self . failUnlessEqual ( children , { } ) )
d1 . addCallback ( lambda res : dnode . has_child ( " edgar " ) )
d1 . addCallback ( lambda answer : self . failUnlessEqual ( answer , False ) )
2007-12-03 21:52:42 +00:00
d1 . addCallback ( lambda res : dnode . set_node ( " see recursive " , dnode , wait_for_numpeers = self . numclients ) )
2007-11-08 11:31:00 +00:00
d1 . addCallback ( lambda res : dnode . has_child ( " see recursive " ) )
d1 . addCallback ( lambda answer : self . failUnlessEqual ( answer , True ) )
2007-11-14 06:08:15 +00:00
d1 . addCallback ( lambda res : dnode . build_manifest ( ) )
d1 . addCallback ( lambda manifest :
self . failUnlessEqual ( len ( manifest ) , 1 ) )
2007-11-08 11:31:00 +00:00
return d1
d . addCallback ( _created_dirnode )
2007-11-07 01:57:11 +00:00
return d
2007-11-29 21:47:35 +00:00
# The default 120 second timeout went off when running it under valgrind
# on my old Windows laptop, so I'm bumping up the timeout.
test_mutable . timeout = 240
2007-11-07 01:57:11 +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
2007-12-03 21:52:42 +00:00
d = self . set_up_nodes ( createprivdir = True )
2007-06-28 18:00:03 +00:00
d . addCallback ( self . log , " starting publish " )
2007-09-25 01:12:37 +00:00
d . addCallback ( self . _do_publish1 )
d . addCallback ( self . _test_runner )
d . addCallback ( self . _do_publish2 )
2007-12-03 21:52:42 +00:00
# at this point, we have the following filesystem (where "R" denotes
# self._root_directory_uri):
# R
# R/subdir1
# R/subdir1/mydata567
# R/subdir1/subdir2/
# R/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 " )
2007-12-03 21:52:42 +00:00
# now we also have (where "P" denotes clients[0]'s automatic private
# dir):
# P/personal/sekrit data
# P/s2-rw -> /subdir1/subdir2/
# P/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-08-22 21:54:34 +00:00
d . addCallback ( self . _test_web_start )
2007-09-26 19:06:55 +00:00
d . addCallback ( self . _test_control )
2007-10-12 02:20:41 +00:00
d . addCallback ( self . _test_cli )
2007-12-04 22:55:27 +00:00
# P now has four top-level children:
# P/personal/sekrit data
# P/s2-ro/
# P/s2-rw/
# P/test_put/ (empty)
2007-10-15 23:16:39 +00:00
d . addCallback ( self . _test_checker )
2007-10-16 19:25:09 +00:00
d . addCallback ( self . _test_verifier )
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-09-25 01:12:37 +00:00
def _do_publish1 ( self , res ) :
2007-06-28 18:00:03 +00:00
ut = upload . Data ( self . data )
c0 = self . clients [ 0 ]
2007-12-03 21:52:42 +00:00
d = c0 . create_empty_dirnode ( wait_for_numpeers = self . numclients )
def _made_root ( new_dirnode ) :
log . msg ( " ZZZ %s -> %s " % ( hasattr ( self , ' _root_directory_uri ' ) and self . _root_directory_uri , new_dirnode . get_uri ( ) , ) )
self . _root_directory_uri = new_dirnode . get_uri ( )
return c0 . create_node_from_uri ( self . _root_directory_uri )
d . addCallback ( _made_root )
d . addCallback ( lambda root : root . create_empty_directory ( " subdir1 " , wait_for_numpeers = self . numclients ) )
2007-06-28 22:05:33 +00:00
def _made_subdir1 ( subdir1_node ) :
2007-09-25 01:12:37 +00:00
self . _subdir1_node = subdir1_node
2007-12-03 21:52:42 +00:00
d1 = subdir1_node . add_file ( " mydata567 " , ut , wait_for_numpeers = self . numclients )
2007-06-28 22:05:33 +00:00
d1 . addCallback ( self . log , " publish finished " )
def _stash_uri ( filenode ) :
self . uri = filenode . get_uri ( )
d1 . addCallback ( _stash_uri )
return d1
d . addCallback ( _made_subdir1 )
2007-06-28 18:00:03 +00:00
return d
2007-09-25 01:12:37 +00:00
def _do_publish2 ( self , res ) :
ut = upload . Data ( self . data )
2007-12-03 21:52:42 +00:00
d = self . _subdir1_node . create_empty_directory ( " subdir2 " , wait_for_numpeers = self . numclients )
d . addCallback ( lambda subdir2 : subdir2 . add_file ( " mydata992 " , ut , wait_for_numpeers = self . numclients ) )
2007-09-25 01:12:37 +00:00
return d
2007-06-28 18:00:03 +00:00
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-12-03 21:52:42 +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-12-03 21:52:42 +00:00
d = self . clients [ 0 ] . get_private_uri ( )
d . addCallback ( self . log , " GOT private directory " )
def _got_root_uri ( privuri ) :
assert privuri
privnode = self . clients [ 0 ] . create_node_from_uri ( privuri )
rootnode = self . clients [ 0 ] . create_node_from_uri ( self . _root_directory_uri )
d1 = privnode . create_empty_directory ( " personal " , wait_for_numpeers = self . numclients )
d1 . addCallback ( self . log , " made P/personal " )
d1 . addCallback ( lambda node : node . add_file ( " sekrit data " , ut , wait_for_numpeers = self . numclients ) )
d1 . addCallback ( self . log , " made P/personal/sekrit data " )
d1 . addCallback ( lambda res : rootnode . get_child_at_path ( [ " subdir1 " , " subdir2 " ] ) )
2007-06-28 22:05:33 +00:00
def _got_s2 ( s2node ) :
2007-12-03 21:52:42 +00:00
d2 = privnode . set_uri ( " s2-rw " , s2node . get_uri ( ) , wait_for_numpeers = self . numclients )
d2 . addCallback ( lambda node : privnode . set_uri ( " s2-ro " , s2node . get_readonly_uri ( ) , wait_for_numpeers = self . numclients ) )
2007-06-28 22:05:33 +00:00
return d2
d1 . addCallback ( _got_s2 )
return d1
2007-12-03 21:52:42 +00:00
d . addCallback ( _got_root_uri )
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 ]
2007-12-03 21:52:42 +00:00
d = defer . succeed ( c1 . create_node_from_uri ( self . _root_directory_uri ) )
2007-06-28 18:00:03 +00:00
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
2007-12-03 21:52:42 +00:00
rootnode = self . clients [ 1 ] . create_node_from_uri ( self . _root_directory_uri )
d = rootnode . get_child_at_path ( " subdir1 " )
2007-06-28 18:00:03 +00:00
d . addCallback ( lambda dirnode :
self . failUnless ( IDirectoryNode . providedBy ( dirnode ) ) )
2007-12-03 21:52:42 +00:00
d . addCallback ( lambda res : rootnode . get_child_at_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 ) )
2007-12-03 21:52:42 +00:00
d . addCallback ( lambda res : rootnode . get_child_at_path ( " subdir1/mydata567 " ) )
2007-06-28 18:00:03 +00:00
def _got_filenode ( filenode ) :
2007-12-03 21:52:42 +00:00
fnode = self . clients [ 1 ] . create_node_from_uri ( filenode . get_uri ( ) )
assert fnode == filenode
2007-06-28 18:00:03 +00:00
d . addCallback ( _got_filenode )
return d
def _check_publish_private ( self , res ) :
# this one uses the path-based API
2007-12-03 21:52:42 +00:00
d = self . clients [ 0 ] . get_private_uri ( )
def _got_private_uri ( privateuri ) :
self . _private_node = self . clients [ 0 ] . create_node_from_uri ( privateuri )
d . addCallback ( _got_private_uri )
d . addCallback ( lambda res : self . _private_node . get_child_at_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-12-03 21:52:42 +00:00
2007-06-28 18:00:03 +00:00
d . addCallback ( lambda dirnode :
2007-12-03 21:52:42 +00:00
self . failUnless ( IDirectoryNode . providedBy ( dirnode ) , dirnode ) )
def get_path ( path ) :
return self . _private_node . get_child_at_path ( path )
d . addCallback ( lambda res : get_path ( " personal/sekrit data " ) )
2007-06-28 18:00:03 +00:00
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-12-03 21:52:42 +00:00
d . addCallback ( lambda res : get_path ( " s2-rw " ) )
2007-06-28 22:05:33 +00:00
d . addCallback ( lambda dirnode : self . failUnless ( dirnode . is_mutable ( ) ) )
2007-12-03 21:52:42 +00:00
d . addCallback ( lambda res : get_path ( " s2-ro " ) )
2007-06-28 22:05:33 +00:00
def _got_s2ro ( dirnode ) :
2007-12-03 21:52:42 +00:00
self . failUnless ( dirnode . is_mutable ( ) , dirnode )
self . failUnless ( dirnode . is_readonly ( ) , dirnode )
2007-06-28 22:05:33 +00:00
d1 = defer . succeed ( None )
d1 . addCallback ( lambda res : dirnode . list ( ) )
d1 . addCallback ( self . log , " dirnode.list " )
2007-12-03 21:52:42 +00:00
d1 . addCallback ( lambda res : self . shouldFail2 ( NotMutableError , " mkdir(nope) " , None , dirnode . create_empty_directory , " nope " ) )
2007-06-28 22:05:33 +00:00
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. " )
2007-12-03 21:52:42 +00:00
d1 . addCallback ( lambda res : self . shouldFail2 ( NotMutableError , " add_file(nope) " , None , dirnode . add_file , " hope " , ut ) )
2007-06-28 22:05:33 +00:00
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) " )
2007-12-03 21:52:42 +00:00
d1 . addCallback ( lambda res : self . shouldFail2 ( NotMutableError , " delete(nope) " , None , dirnode . delete , " mydata992 " ) )
2007-06-28 22:05:33 +00:00
2007-12-03 21:52:42 +00:00
d1 . addCallback ( lambda res : self . shouldFail2 ( NotMutableError , " set_uri(nope) " , None , dirnode . set_uri , " hopeless " , self . uri ) )
2007-06-28 22:05:33 +00:00
2007-12-03 21:52:42 +00:00
d1 . addCallback ( lambda res : self . shouldFail2 ( KeyError , " get(missing) " , " ' missing ' " , dirnode . get , " missing " ) )
2007-06-28 22:05:33 +00:00
personal = self . _personal_node
2007-12-03 21:52:42 +00:00
d1 . addCallback ( lambda res : self . shouldFail2 ( NotMutableError , " mv from readonly " , None , dirnode . move_child_to , " mydata992 " , personal , " nope " ) )
2007-06-28 22:05:33 +00:00
d1 . addCallback ( self . log , " doing move_child_to(ro)2 " )
2007-12-03 21:52:42 +00:00
d1 . addCallback ( lambda res : self . shouldFail2 ( NotMutableError , " mv to readonly " , None , personal . move_child_to , " sekrit data " , dirnode , " nope " ) )
2007-06-28 22:05:33 +00:00
d1 . addCallback ( self . log , " finished with _got_s2ro " )
return d1
d . addCallback ( _got_s2ro )
2007-12-03 21:52:42 +00:00
def _got_home ( dummy ) :
home = self . _private_node
2007-06-28 22:05:33 +00:00
personal = self . _personal_node
d1 = defer . succeed ( None )
2007-12-03 21:52:42 +00:00
d1 . addCallback ( self . log , " mv ' P/personal/sekrit data ' to P/sekrit " )
2007-06-28 22:05:33 +00:00
d1 . addCallback ( lambda res :
personal . move_child_to ( " sekrit data " , home , " sekrit " ) )
2007-12-03 21:52:42 +00:00
d1 . addCallback ( self . log , " mv P/sekrit ' P/sekrit data ' " )
2007-06-28 22:05:33 +00:00
d1 . addCallback ( lambda res :
home . move_child_to ( " sekrit " , home , " sekrit data " ) )
2007-12-03 21:52:42 +00:00
d1 . addCallback ( self . log , " mv ' P/sekret data ' P/personal/ " )
2007-06-28 22:05:33 +00:00
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:
2007-12-03 21:52:42 +00:00
# P/personal/
# P/personal/sekrit data
# P/s2-rw (same as P/s2-ro)
# P/s2-rw/mydata992 (same as P/s2-rw/mydata992)
2007-06-28 22:05:33 +00:00
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 ) )
2007-12-03 21:52:42 +00:00
def shouldFail2 ( self , expected_failure , which , substring , callable , * args , * * kwargs ) :
assert substring is None or isinstance ( substring , str )
d = defer . maybeDeferred ( callable , * args , * * kwargs )
def done ( res ) :
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 ) )
d . addBoth ( done )
return d
2007-08-16 23:49:40 +00:00
def PUT ( self , urlpath , data ) :
url = self . webish_url + urlpath
return getPage ( url , method = " PUT " , postdata = data )
def GET ( self , urlpath , followRedirect = False ) :
url = self . webish_url + urlpath
return getPage ( url , method = " GET " , followRedirect = followRedirect )
2006-12-07 19:48:06 +00:00
def _test_web ( self , res ) :
base = self . webish_url
2007-12-03 21:52:42 +00:00
public = " uri/ " + self . _root_directory_uri . replace ( " / " , " ! " )
2006-12-07 19:48:06 +00:00
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-08-12 17:29:38 +00:00
expected = " My nodeid: <span> %s </span> " % ( b32encode ( self . clients [ 0 ] . nodeid ) . lower ( ) , )
2007-03-29 21:31:55 +00:00
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-08-16 23:49:40 +00:00
d . addCallback ( self . log , " done with _got_welcome " )
2007-12-03 21:52:42 +00:00
d . addCallback ( lambda res : getPage ( base + public ) )
d . addCallback ( lambda res : getPage ( base + public + " /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-08-16 23:49:40 +00:00
d . addCallback ( self . log , " done with _got_subdir1 " )
2007-06-15 08:38:55 +00:00
d . addCallback ( lambda res :
2007-12-03 21:52:42 +00:00
getPage ( base + public + " /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
2007-08-16 23:49:40 +00:00
d . addCallback ( self . log , " _get_from_uri " )
2007-04-24 08:41:54 +00:00
def _get_from_uri ( res ) :
2007-08-16 23:49:40 +00:00
return getPage ( base + " uri/ %s ?filename= %s "
2007-04-24 08:41:54 +00:00
% ( 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
2007-08-16 23:49:40 +00:00
d . addCallback ( self . log , " _get_from_uri2 " )
2007-04-24 08:41:54 +00:00
def _get_from_uri2 ( res ) :
2007-08-16 23:49:40 +00:00
return getPage ( base + " uri?uri= %s " % ( self . uri , ) )
2007-04-24 08:41:54 +00:00
d . addCallback ( _get_from_uri2 )
2007-12-03 21:52:42 +00:00
d . addCallback ( _got_from_uri )
2007-04-24 08:41:54 +00:00
2007-07-03 20:18:14 +00:00
# download from a bogus URI, make sure we get a reasonable error
2007-08-16 23:49:40 +00:00
d . addCallback ( self . log , " _get_from_bogus_uri " )
2007-07-03 20:18:14 +00:00
def _get_from_bogus_uri ( res ) :
2007-08-16 23:49:40 +00:00
d1 = getPage ( base + " uri/ %s ?filename= %s "
% ( self . mangle_uri ( self . uri ) , " mydata567 " ) )
d1 . addBoth ( self . shouldFail , Error , " downloading bogus URI " ,
2007-08-17 00:05:26 +00:00
" 410 " )
2007-08-16 23:49:40 +00:00
return d1
2007-07-03 20:18:14 +00:00
d . addCallback ( _get_from_bogus_uri )
2007-08-16 23:49:40 +00:00
# upload a file with PUT
d . addCallback ( self . log , " about to try PUT " )
2007-12-03 21:52:42 +00:00
d . addCallback ( lambda res : self . PUT ( public + " /subdir3/new.txt " ,
2007-08-16 23:49:40 +00:00
" new.txt contents " ) )
2007-12-03 21:52:42 +00:00
d . addCallback ( lambda res : self . GET ( public + " /subdir3/new.txt " ) )
2007-08-16 23:49:40 +00:00
d . addCallback ( self . failUnlessEqual , " new.txt contents " )
2007-09-19 08:43:44 +00:00
# and again with something large enough to use multiple segments,
# and hopefully trigger pauseProducing too
2007-12-03 21:52:42 +00:00
d . addCallback ( lambda res : self . PUT ( public + " /subdir3/big.txt " ,
2007-09-19 08:43:44 +00:00
" big " * 500000 ) ) # 1.5MB
2007-12-03 21:52:42 +00:00
d . addCallback ( lambda res : self . GET ( public + " /subdir3/big.txt " ) )
2007-09-19 08:43:44 +00:00
d . addCallback ( lambda res : self . failUnlessEqual ( len ( res ) , 1500000 ) )
2007-07-03 20:47:37 +00:00
2007-08-17 00:03:50 +00:00
# can we replace files in place?
2007-12-03 21:52:42 +00:00
d . addCallback ( lambda res : self . PUT ( public + " /subdir3/new.txt " ,
2007-08-17 00:03:50 +00:00
" NEWER contents " ) )
2007-12-03 21:52:42 +00:00
d . addCallback ( lambda res : self . GET ( public + " /subdir3/new.txt " ) )
2007-08-17 00:03:50 +00:00
d . addCallback ( self . failUnlessEqual , " NEWER contents " )
2007-08-22 21:54:34 +00:00
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-12-03 21:52:42 +00:00
# url = base + "somedir/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-08-22 21:54:34 +00:00
def _test_web_start ( self , res ) :
basedir = self . clients [ 0 ] . basedir
startfile = os . path . join ( basedir , " start.html " )
self . failUnless ( os . path . exists ( startfile ) )
start_html = open ( startfile , " r " ) . read ( )
self . failUnless ( self . webish_url in start_html )
2007-12-03 21:52:42 +00:00
d = self . clients [ 0 ] . get_private_uri ( )
def done ( private_uri ) :
private_url = self . webish_url + " uri/ " + private_uri . replace ( " / " , " ! " )
self . failUnless ( private_url in start_html )
d . addCallback ( done )
return d
2007-08-22 21:54:34 +00:00
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 )
2007-08-28 06:41:40 +00:00
if pieces [ - 3 ] == " storage " and pieces [ - 2 ] == " shares " :
# we're sitting in .../storage/shares/$SINDEX , and there are
2007-07-13 23:58:08 +00:00
# sharefiles here
filename = os . path . join ( dirpath , filenames [ 0 ] )
2007-12-03 21:52:42 +00:00
# peek at the magic to see if it is a chk share
magic = open ( filename , " rb " ) . read ( 4 )
if magic == ' \x00 \x00 \x00 \x01 ' :
break
2007-06-26 22:55:00 +00:00
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 ( )
2007-09-02 21:48:20 +00:00
rc = runner . runner ( [ " dump-share " ,
2007-06-26 23:51:38 +00:00
filename ] ,
stdout = out , stderr = err )
output = out . getvalue ( )
2007-06-26 22:55:00 +00:00
self . failUnlessEqual ( rc , 0 )
2007-09-25 01:12:37 +00:00
# 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-12-03 21:52:42 +00:00
self . failUnless ( " segment_size: %d \n " % mathutil . next_multiple ( len ( self . data ) , 3 ) in output )
2007-09-25 01:12:37 +00:00
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 " ,
2007-12-03 21:52:42 +00:00
" share_root_hash " , ) :
2007-06-26 22:55:00 +00:00
self . failUnless ( " %s : " % key in output , key )
2007-09-26 19:06:55 +00:00
def _test_control ( self , res ) :
# exercise the remote-control-the-client foolscap interfaces in
# allmydata.control (mostly used for performance tests)
c0 = self . clients [ 0 ]
control_furl_file = os . path . join ( c0 . basedir , " control.furl " )
control_furl = open ( control_furl_file , " r " ) . read ( ) . strip ( )
# it doesn't really matter which Tub we use to connect to the client,
2007-12-03 21:52:42 +00:00
# so let's just use our IntroducerNode's
d = self . introducer . tub . getReference ( control_furl )
2007-09-26 19:06:55 +00:00
d . addCallback ( self . _test_control2 , control_furl_file )
return d
def _test_control2 ( self , rref , filename ) :
d = rref . callRemote ( " upload_from_file_to_uri " , filename )
downfile = os . path . join ( self . basedir , " control.downfile " )
d . addCallback ( lambda uri :
rref . callRemote ( " download_from_uri_to_file " ,
uri , downfile ) )
def _check ( res ) :
self . failUnlessEqual ( res , downfile )
data = open ( downfile , " r " ) . read ( )
expected_data = open ( filename , " r " ) . read ( )
self . failUnlessEqual ( data , expected_data )
d . addCallback ( _check )
d . addCallback ( lambda res : rref . callRemote ( " speed_test " , 1 , 200 ) )
if sys . platform == " linux2 " :
d . addCallback ( lambda res : rref . callRemote ( " get_memory_usage " ) )
2007-09-26 19:21:15 +00:00
d . addCallback ( lambda res : rref . callRemote ( " measure_peer_response_time " ) )
2007-09-26 19:06:55 +00:00
return d
2007-10-12 02:20:41 +00:00
def _test_cli ( self , res ) :
# run various CLI commands (in a thread, since they use blocking
# network calls)
2007-12-03 21:52:42 +00:00
private_uri = self . _private_node . get_uri ( )
some_uri = self . _root_directory_uri
2007-10-12 02:20:41 +00:00
nodeargs = [
" --node-url " , self . webish_url ,
" --root-uri " , private_uri ,
]
2007-10-21 19:33:17 +00:00
public_nodeargs = [
" --node-url " , self . webish_url ,
2007-12-03 21:52:42 +00:00
" --root-uri " , some_uri ,
2007-10-21 19:33:17 +00:00
]
TESTDATA = " I will not write the same thing over and over. \n " * 100
d = defer . succeed ( None )
2007-10-12 02:20:41 +00:00
2007-10-21 19:33:17 +00:00
def _ls_root ( res ) :
argv = [ " ls " ] + nodeargs
return self . _run_cli ( argv )
d . addCallback ( _ls_root )
def _check_ls_root ( ( out , err ) ) :
2007-10-12 02:20:41 +00:00
self . failUnless ( " personal " in out )
self . failUnless ( " s2-ro " in out )
self . failUnless ( " s2-rw " in out )
self . failUnlessEqual ( err , " " )
2007-10-21 19:33:17 +00:00
d . addCallback ( _check_ls_root )
def _ls_subdir ( res ) :
argv = [ " ls " ] + nodeargs + [ " personal " ]
return self . _run_cli ( argv )
d . addCallback ( _ls_subdir )
def _check_ls_subdir ( ( out , err ) ) :
self . failUnless ( " sekrit data " in out )
self . failUnlessEqual ( err , " " )
d . addCallback ( _check_ls_subdir )
def _ls_public_subdir ( res ) :
argv = [ " ls " ] + public_nodeargs + [ " subdir1 " ]
return self . _run_cli ( argv )
d . addCallback ( _ls_public_subdir )
def _check_ls_public_subdir ( ( out , err ) ) :
self . failUnless ( " subdir2 " in out )
self . failUnless ( " mydata567 " in out )
self . failUnlessEqual ( err , " " )
d . addCallback ( _check_ls_public_subdir )
def _ls_file ( res ) :
argv = [ " ls " ] + public_nodeargs + [ " subdir1/mydata567 " ]
return self . _run_cli ( argv )
d . addCallback ( _ls_file )
def _check_ls_file ( ( out , err ) ) :
self . failUnlessEqual ( out . strip ( ) , " 112 subdir1/mydata567 " )
self . failUnlessEqual ( err , " " )
d . addCallback ( _check_ls_file )
# tahoe_ls doesn't currently handle the error correctly: it tries to
# JSON-parse a traceback.
## def _ls_missing(res):
## argv = ["ls"] + nodeargs + ["bogus"]
## return self._run_cli(argv)
## d.addCallback(_ls_missing)
## def _check_ls_missing((out,err)):
## print "OUT", out
## print "ERR", err
## self.failUnlessEqual(err, "")
## d.addCallback(_check_ls_missing)
2007-10-12 02:20:41 +00:00
def _put ( res ) :
tdir = self . getdir ( " cli_put " )
fileutil . make_dirs ( tdir )
fn = os . path . join ( tdir , " upload_me " )
2007-12-12 02:16:33 +00:00
f = open ( fn , " wb " )
2007-10-21 19:33:17 +00:00
f . write ( TESTDATA )
2007-10-12 02:20:41 +00:00
f . close ( )
argv = [ " put " ] + nodeargs + [ fn , " test_put/upload.txt " ]
return self . _run_cli ( argv )
d . addCallback ( _put )
def _check_put ( ( out , err ) ) :
self . failUnless ( " 200 OK " in out )
self . failUnlessEqual ( err , " " )
2007-12-03 21:52:42 +00:00
d = self . _private_node . get_child_at_path ( " test_put/upload.txt " )
2007-10-12 02:20:41 +00:00
d . addCallback ( lambda filenode : filenode . download_to_data ( ) )
def _check_put2 ( res ) :
2007-10-21 19:33:17 +00:00
self . failUnlessEqual ( res , TESTDATA )
2007-10-12 02:20:41 +00:00
d . addCallback ( _check_put2 )
return d
d . addCallback ( _check_put )
2007-10-21 19:33:17 +00:00
def _get_to_stdout ( res ) :
2007-10-12 02:20:41 +00:00
argv = [ " get " ] + nodeargs + [ " test_put/upload.txt " ]
return self . _run_cli ( argv )
2007-10-21 19:33:17 +00:00
d . addCallback ( _get_to_stdout )
def _check_get_to_stdout ( ( out , err ) ) :
self . failUnlessEqual ( out , TESTDATA )
2007-10-12 02:20:41 +00:00
self . failUnlessEqual ( err , " " )
2007-10-21 19:33:17 +00:00
d . addCallback ( _check_get_to_stdout )
get_to_file_target = os . path . join ( self . basedir , " get.downfile " )
def _get_to_file ( res ) :
argv = [ " get " ] + nodeargs + [ " test_put/upload.txt " ,
get_to_file_target ]
return self . _run_cli ( argv )
d . addCallback ( _get_to_file )
def _check_get_to_file ( ( out , err ) ) :
data = open ( get_to_file_target , " rb " ) . read ( )
self . failUnlessEqual ( data , TESTDATA )
self . failUnlessEqual ( out , " " )
self . failUnlessEqual ( err , " test_put/upload.txt retrieved and written to system/SystemTest/test_vdrive/get.downfile \n " )
d . addCallback ( _check_get_to_file )
2007-10-12 02:20:41 +00:00
2007-10-12 03:31:48 +00:00
def _mv ( res ) :
argv = [ " mv " ] + nodeargs + [ " test_put/upload.txt " ,
" test_put/moved.txt " ]
return self . _run_cli ( argv )
d . addCallback ( _mv )
def _check_mv ( ( out , err ) ) :
self . failUnless ( " OK " in out )
self . failUnlessEqual ( err , " " )
2007-12-03 21:52:42 +00:00
d = self . shouldFail2 ( KeyError , " test_cli._check_rm " , " ' upload.txt ' " , self . _private_node . get_child_at_path , " test_put/upload.txt " )
2007-10-12 03:31:48 +00:00
d . addCallback ( lambda res :
2007-12-03 21:52:42 +00:00
self . _private_node . get_child_at_path ( " test_put/moved.txt " ) )
2007-10-12 03:31:48 +00:00
d . addCallback ( lambda filenode : filenode . download_to_data ( ) )
def _check_mv2 ( res ) :
2007-10-21 19:33:17 +00:00
self . failUnlessEqual ( res , TESTDATA )
2007-10-12 03:31:48 +00:00
d . addCallback ( _check_mv2 )
return d
d . addCallback ( _check_mv )
2007-10-12 02:20:41 +00:00
def _rm ( res ) :
2007-10-12 03:31:48 +00:00
argv = [ " rm " ] + nodeargs + [ " test_put/moved.txt " ]
2007-10-12 02:20:41 +00:00
return self . _run_cli ( argv )
d . addCallback ( _rm )
def _check_rm ( ( out , err ) ) :
self . failUnless ( " 200 OK " in out )
self . failUnlessEqual ( err , " " )
2007-12-03 21:52:42 +00:00
d = self . shouldFail2 ( KeyError , " test_cli._check_rm " , " ' moved.txt ' " , self . _private_node . get_child_at_path , " test_put/moved.txt " )
2007-10-12 02:20:41 +00:00
return d
d . addCallback ( _check_rm )
return d
def _run_cli ( self , argv ) :
stdout , stderr = StringIO ( ) , StringIO ( )
d = threads . deferToThread ( runner . runner , argv , run_by_human = False ,
stdout = stdout , stderr = stderr )
def _done ( res ) :
return stdout . getvalue ( ) , stderr . getvalue ( )
d . addCallback ( _done )
return d
2007-10-15 23:16:39 +00:00
def _test_checker ( self , res ) :
2007-12-03 21:52:42 +00:00
d = self . _private_node . build_manifest ( )
2007-10-23 00:46:24 +00:00
d . addCallback ( self . _test_checker_2 )
return d
def _test_checker_2 ( self , manifest ) :
checker1 = self . clients [ 1 ] . getServiceNamed ( " checker " )
2007-12-05 01:00:12 +00:00
self . failUnlessEqual ( checker1 . checker_results_for ( None ) , [ ] )
self . failUnlessEqual ( checker1 . checker_results_for ( list ( manifest ) [ 0 ] ) ,
[ ] )
2007-10-23 00:46:24 +00:00
dl = [ ]
starting_time = time . time ( )
for si in manifest :
dl . append ( checker1 . check ( si ) )
d = deferredutil . DeferredListShouldSucceed ( dl )
def _check_checker_results ( res ) :
2007-10-15 23:16:39 +00:00
for i in res :
2007-10-17 09:25:50 +00:00
if type ( i ) is bool :
self . failUnless ( i is True )
else :
( needed , total , found , sharemap ) = i
self . failUnlessEqual ( needed , 3 )
self . failUnlessEqual ( total , 10 )
self . failUnlessEqual ( found , total )
self . failUnlessEqual ( len ( sharemap . keys ( ) ) , 10 )
peers = set ( )
for shpeers in sharemap . values ( ) :
peers . update ( shpeers )
self . failUnlessEqual ( len ( peers ) , self . numclients - 1 )
2007-10-23 00:46:24 +00:00
d . addCallback ( _check_checker_results )
def _check_stored_results ( res ) :
finish_time = time . time ( )
all_results = [ ]
for si in manifest :
results = checker1 . checker_results_for ( si )
2007-12-03 21:52:42 +00:00
if not results :
# TODO: implement checker for mutable files and implement tests of that checker
continue
2007-10-23 00:46:24 +00:00
self . failUnlessEqual ( len ( results ) , 1 )
when , those_results = results [ 0 ]
self . failUnless ( isinstance ( when , ( int , float ) ) )
self . failUnless ( starting_time < = when < = finish_time )
all_results . append ( those_results )
_check_checker_results ( all_results )
d . addCallback ( _check_stored_results )
2007-12-04 22:55:27 +00:00
d . addCallback ( self . _test_checker_3 )
return d
def _test_checker_3 ( self , res ) :
# check one file, through FileNode.check()
d = self . _private_node . get_child_at_path ( " personal/sekrit data " )
d . addCallback ( lambda n : n . check ( ) )
def _checked ( results ) :
# 'sekrit data' is small, and fits in a LiteralFileNode, so
# checking it is trivial and always returns True
self . failUnlessEqual ( results , True )
d . addCallback ( _checked )
c0 = self . clients [ 1 ]
n = c0 . create_node_from_uri ( self . _root_directory_uri )
d . addCallback ( lambda res : n . get_child_at_path ( " subdir1/mydata567 " ) )
d . addCallback ( lambda n : n . check ( ) )
def _checked2 ( results ) :
# mydata567 is large and lives in a CHK
( needed , total , found , sharemap ) = results
self . failUnlessEqual ( needed , 3 )
self . failUnlessEqual ( total , 10 )
self . failUnlessEqual ( found , 10 )
self . failUnlessEqual ( len ( sharemap ) , 10 )
for shnum in range ( 10 ) :
self . failUnlessEqual ( len ( sharemap [ shnum ] ) , 1 )
d . addCallback ( _checked2 )
2007-10-15 23:16:39 +00:00
return d
2007-12-04 22:55:27 +00:00
2007-10-16 19:25:09 +00:00
def _test_verifier ( self , res ) :
checker1 = self . clients [ 1 ] . getServiceNamed ( " checker " )
2007-12-03 21:52:42 +00:00
d = self . _private_node . build_manifest ( )
2007-10-16 19:25:09 +00:00
def _check_all ( manifest ) :
dl = [ ]
for si in manifest :
dl . append ( checker1 . verify ( si ) )
return deferredutil . DeferredListShouldSucceed ( dl )
d . addCallback ( _check_all )
def _done ( res ) :
for i in res :
self . failUnless ( i is True )
d . addCallback ( _done )
2007-12-05 01:00:12 +00:00
d . addCallback ( lambda res : checker1 . verify ( None ) )
d . addCallback ( self . failUnlessEqual , True )
2007-10-16 19:25:09 +00:00
return d