2010-07-11 20:02:52 +00:00
2010-08-09 22:03:42 +00:00
import os . path , re , urllib , time
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
Change OphandleTable to use a deterministic clock, so we can test it
To test the changes for #577, we need a deterministic way to simulate
the passage of long periods of time. twisted.internet.task.Clock seems,
from my Googling, to be the way to go for this functionality. I changed
a few things so that OphandleTable would use twisted.internet.task.Clock
when testing:
* WebishServer.__init___ now takes an optional 'clock' parameter,
* which it passes to the root.Root instance it creates.
* root.Root.__init__ now takes an optional 'clock' parameter, which it
passes to the OphandleTable.__init__ method.
* OphandleTable.__init__ now takes an optional 'clock' parameter. If
it is provided, and it isn't None, its callLater method will be used
to schedule ophandle expirations (as opposed to using
reactor.callLater, which is what OphandleTable does normally).
* The WebMixin object in test_web.py now sets a self.clock parameter,
which is a twisted.internet.task.Clock that it feeds to the
WebishServer it creates.
Tests using the WebMixin can control the passage of time in
OphandleTable by accessing self.clock.
2010-02-20 21:07:13 +00:00
from twisted . internet . task import Clock
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
2010-01-27 06:44:30 +00:00
from allmydata import interfaces , uri , webish , dirnode
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
2010-08-04 07:27:10 +00:00
from allmydata . immutable import upload
from allmydata . immutable . downloader . status import DownloadStatus
2009-10-13 02:34:44 +00:00
from allmydata . dirnode import DirectoryNode
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
from allmydata . nodemaker import NodeMaker
2009-07-03 01:07:49 +00:00
from allmydata . unknown import UnknownNode
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
2011-06-29 22:25:55 +00:00
from allmydata . util import fileutil , base32 , hashutil
2009-12-01 22:44:35 +00:00
from allmydata . util . consumer import download_to_data
2010-01-27 06:44:30 +00:00
from allmydata . util . netstring import split_netstring
2010-07-18 14:29:15 +00:00
from allmydata . util . encodingutil import to_str
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
from allmydata . test . common import FakeCHKFileNode , FakeMutableFileNode , \
2009-10-13 02:34:44 +00:00
create_chk_filenode , WebErrorMixin , ShouldFailMixin , make_mutable_file_uri
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
from allmydata . interfaces import IMutableFileNode
2008-04-17 20:02:22 +00:00
from allmydata . mutable import servermap , publish , retrieve
2010-02-26 08:14:33 +00:00
import allmydata . test . common_util as testutil
2009-02-17 05:12:42 +00:00
from allmydata . test . no_network import GridTestMixin
2009-02-25 01:33:00 +00:00
from allmydata . test . common_web import HTTPClientGETFactory , \
HTTPClientHEADFactory
2009-11-18 01:54:44 +00:00
from allmydata . client import Client , SecretHolder
2009-02-25 01:33:00 +00:00
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
2010-07-11 20:02:52 +00:00
unknown_rwcap = u " lafs://from_the_future_rw_ \u263A " . encode ( ' utf-8 ' )
unknown_rocap = u " ro.lafs://readonly_from_the_future_ro_ \u263A " . encode ( ' utf-8 ' )
unknown_immcap = u " imm.lafs://immutable_from_the_future_imm_ \u263A " . encode ( ' utf-8 ' )
2010-05-19 05:51:46 +00:00
2009-02-20 19:15:54 +00:00
class FakeStatsProvider :
def get_stats ( self ) :
stats = { ' stats ' : { } , ' counters ' : { } }
return stats
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
class FakeNodeMaker ( NodeMaker ) :
def _create_lit ( self , cap ) :
return FakeCHKFileNode ( cap )
def _create_immutable ( self , cap ) :
return FakeCHKFileNode ( cap )
def _create_mutable ( self , cap ) :
2009-11-11 22:25:42 +00:00
return FakeMutableFileNode ( None , None , None , None ) . init_from_cap ( cap )
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
def create_mutable_file ( self , contents = " " , keysize = None ) :
n = FakeMutableFileNode ( None , None , None , None )
return n . create ( contents )
2009-10-12 22:28:08 +00:00
class FakeUploader ( service . Service ) :
name = " uploader "
def upload ( self , uploadable , history = None ) :
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
d = uploadable . get_size ( )
d . addCallback ( lambda size : uploadable . read ( size ) )
def _got_data ( datav ) :
data = " " . join ( datav )
n = create_chk_filenode ( data )
results = upload . UploadResults ( )
results . uri = n . get_uri ( )
return results
d . addCallback ( _got_data )
return d
2009-10-12 22:28:08 +00:00
def get_helper_info ( self ) :
return ( None , False )
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
2011-08-01 18:54:01 +00:00
class FakeIServer :
def __init__ ( self , binaryserverid ) :
self . binaryserverid = binaryserverid
def get_name ( self ) : return " short "
def get_longname ( self ) : return " long "
def get_serverid ( self ) : return self . binaryserverid
2010-08-09 22:03:42 +00:00
def build_one_ds ( ) :
ds = DownloadStatus ( " storage_index " , 1234 )
now = time . time ( )
2011-08-01 18:54:01 +00:00
serverA = FakeIServer ( hashutil . tagged_hash ( " foo " , " serverid_a " ) [ : 20 ] )
serverB = FakeIServer ( hashutil . tagged_hash ( " foo " , " serverid_a " ) [ : 20 ] )
2011-06-29 22:25:55 +00:00
storage_index = hashutil . storage_index_hash ( " SI " )
e0 = ds . add_segment_request ( 0 , now )
e0 . activate ( now + 0.5 )
e0 . deliver ( now + 1 , 0 , 100 , 0.5 ) # when, start,len, decodetime
e1 = ds . add_segment_request ( 1 , now + 2 )
e1 . error ( now + 3 )
2010-08-10 06:06:03 +00:00
# two outstanding requests
2011-06-29 22:25:55 +00:00
e2 = ds . add_segment_request ( 2 , now + 4 )
e3 = ds . add_segment_request ( 3 , now + 5 )
del e2 , e3 # hush pyflakes
2010-08-09 22:03:42 +00:00
2010-08-15 14:19:33 +00:00
# simulate a segment which gets delivered faster than a system clock tick (ticket #1166)
2011-06-29 22:25:55 +00:00
e = ds . add_segment_request ( 4 , now )
e . activate ( now )
e . deliver ( now , 0 , 140 , 0.5 )
2010-08-15 14:19:33 +00:00
2011-08-01 18:54:01 +00:00
e = ds . add_dyhb_request ( serverA , now )
2010-08-09 22:03:42 +00:00
e . finished ( [ 1 , 2 ] , now + 1 )
2011-08-01 18:54:01 +00:00
e = ds . add_dyhb_request ( serverB , now + 2 ) # left unfinished
2010-08-09 22:03:42 +00:00
e = ds . add_read_event ( 0 , 120 , now )
e . update ( 60 , 0.5 , 0.1 ) # bytes, decrypttime, pausetime
e . finished ( now + 1 )
e = ds . add_read_event ( 120 , 30 , now + 2 ) # left unfinished
2011-08-01 18:54:01 +00:00
e = ds . add_block_request ( serverA , 1 , 100 , 20 , now )
2010-08-09 22:03:42 +00:00
e . finished ( 20 , now + 1 )
2011-08-01 18:54:01 +00:00
e = ds . add_block_request ( serverB , 1 , 120 , 30 , now + 1 ) # left unfinished
2010-08-09 22:03:42 +00:00
# make sure that add_read_event() can come first too
2011-06-29 22:25:55 +00:00
ds1 = DownloadStatus ( storage_index , 1234 )
2010-08-09 22:03:42 +00:00
e = ds1 . add_read_event ( 0 , 120 , now )
e . update ( 60 , 0.5 , 0.1 ) # bytes, decrypttime, pausetime
e . finished ( now + 1 )
return ds
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
class FakeHistory :
_all_upload_status = [ upload . UploadStatus ( ) ]
2010-08-09 22:03:42 +00:00
_all_download_status = [ build_one_ds ( ) ]
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
_all_mapupdate_statuses = [ servermap . UpdateStatus ( ) ]
_all_publish_statuses = [ publish . PublishStatus ( ) ]
_all_retrieve_statuses = [ retrieve . RetrieveStatus ( ) ]
def list_all_upload_statuses ( self ) :
return self . _all_upload_status
def list_all_download_statuses ( self ) :
return self . _all_download_status
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 ) :
return [ ]
2009-10-12 22:28:08 +00:00
class FakeClient ( Client ) :
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
def __init__ ( self ) :
2009-10-12 22:28:08 +00:00
# don't upcall to Client.__init__, since we only want to initialize a
# minimal subset
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
service . MultiService . __init__ ( self )
2009-10-12 22:28:08 +00:00
self . nodeid = " fake_nodeid "
self . nickname = " fake_nickname "
self . introducer_furl = " None "
self . stats_provider = FakeStatsProvider ( )
2009-11-18 01:54:44 +00:00
self . _secret_holder = SecretHolder ( " lease secret " , " convergence secret " )
2009-10-12 22:28:08 +00:00
self . helper = None
self . convergence = " some random string "
self . storage_broker = StorageFarmBroker ( None , permute_peers = True )
self . introducer_client = None
self . history = FakeHistory ( )
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
self . uploader = FakeUploader ( )
2009-10-12 22:28:08 +00:00
self . uploader . setServiceParent ( self )
2009-11-18 01:54:44 +00:00
self . nodemaker = FakeNodeMaker ( None , self . _secret_holder , None ,
2010-08-04 07:27:10 +00:00
self . uploader , None ,
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
None , None )
2008-03-01 06:03:00 +00:00
2009-10-12 22:28:08 +00:00
def startService ( self ) :
return service . MultiService . startService ( self )
def stopService ( self ) :
return service . MultiService . stopService ( self )
2007-12-03 21:52:42 +00:00
2008-06-03 07:03:16 +00:00
MUTABLE_SIZELIMIT = FakeMutableFileNode . MUTABLE_SIZELIMIT
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 ( )
Change OphandleTable to use a deterministic clock, so we can test it
To test the changes for #577, we need a deterministic way to simulate
the passage of long periods of time. twisted.internet.task.Clock seems,
from my Googling, to be the way to go for this functionality. I changed
a few things so that OphandleTable would use twisted.internet.task.Clock
when testing:
* WebishServer.__init___ now takes an optional 'clock' parameter,
* which it passes to the root.Root instance it creates.
* root.Root.__init__ now takes an optional 'clock' parameter, which it
passes to the OphandleTable.__init__ method.
* OphandleTable.__init__ now takes an optional 'clock' parameter. If
it is provided, and it isn't None, its callLater method will be used
to schedule ophandle expirations (as opposed to using
reactor.callLater, which is what OphandleTable does normally).
* The WebMixin object in test_web.py now sets a self.clock parameter,
which is a twisted.internet.task.Clock that it feeds to the
WebishServer it creates.
Tests using the WebMixin can control the passage of time in
OphandleTable by accessing self.clock.
2010-02-20 21:07:13 +00:00
self . clock = Clock ( )
self . ws = webish . WebishServer ( self . s , " 0 " , staticdir = self . staticdir ,
clock = self . clock )
2009-10-12 22:28:08 +00:00
self . ws . setServiceParent ( self . s )
2011-01-17 07:47:51 +00:00
self . webish_port = self . ws . getPortnum ( )
self . webish_url = self . ws . getURL ( )
assert self . webish_url . endswith ( " / " )
self . webish_url = self . webish_url [ : - 1 ] # these tests add their own /
2007-07-07 02:43:55 +00:00
2009-10-12 22:45:06 +00:00
l = [ self . s . create_dirnode ( ) for x in range ( 6 ) ]
2007-12-03 21:52:42 +00:00
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
2009-10-12 23:51:26 +00:00
self . public_root . set_uri ( u " foo " , foo . get_uri ( ) ,
foo . get_readonly_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 )
2009-10-12 23:51:26 +00:00
foo . set_uri ( u " bar.txt " , self . _bar_txt_uri , 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
2009-10-12 23:51:26 +00:00
foo . set_uri ( u " empty " , res [ 3 ] [ 1 ] . get_uri ( ) ,
res [ 3 ] [ 1 ] . get_readonly_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
2009-10-12 23:51:26 +00:00
foo . set_uri ( u " sub " , sub_uri , 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 )
2009-10-12 23:51:26 +00:00
foo . set_uri ( u " blockingfile " , blocking_uri , blocking_uri )
2008-02-14 22:45:56 +00:00
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
2009-10-12 23:51:26 +00:00
foo . set_uri ( unicode_filename , self . _bar_txt_uri , 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
2009-10-12 23:51:26 +00:00
sub . set_uri ( u " baz.txt " , baz_file , 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 ]
2009-10-12 23:51:26 +00:00
self . public_root . set_uri ( u " reedownlee " , rodir . get_readonly_uri ( ) ,
rodir . get_readonly_uri ( ) )
rodir . set_uri ( u " nor " , baz_file , 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
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
n = create_chk_filenode ( contents )
2007-12-05 06:01:37 +00:00
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 ) :
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( 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 ) )
2010-07-18 14:29:15 +00:00
self . failUnlessEqual ( data [ 0 ] , " 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
2010-07-18 14:29:15 +00:00
self . failUnlessReallyEqual ( to_str ( data [ 1 ] [ " ro_uri " ] ) , self . _bar_txt_uri )
self . failUnlessReallyEqual ( to_str ( data [ 1 ] [ " verify_uri " ] ) , self . _bar_txt_verifycap )
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( 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 ) )
2010-07-18 14:29:15 +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
2010-07-18 14:29:15 +00:00
self . failUnlessReallyEqual ( to_str ( data [ 1 ] [ " rw_uri " ] ) , self . _foo_uri )
self . failUnlessReallyEqual ( to_str ( data [ 1 ] [ " ro_uri " ] ) , self . _foo_readonly_uri )
self . failUnlessReallyEqual ( to_str ( 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 " ] ] )
2010-07-18 14:29:15 +00:00
self . failUnlessEqual ( kidnames ,
[ 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 ( ) ] )
2010-07-18 14:29:15 +00:00
self . failUnlessEqual ( kids [ u " sub " ] [ 0 ] , " dirnode " )
2010-06-18 23:01:19 +00:00
self . failUnlessIn ( " metadata " , kids [ u " sub " ] [ 1 ] )
self . failUnlessIn ( " tahoe " , kids [ u " sub " ] [ 1 ] [ " metadata " ] )
tahoe_md = kids [ u " sub " ] [ 1 ] [ " metadata " ] [ " tahoe " ]
self . failUnlessIn ( " linkcrtime " , tahoe_md )
self . failUnlessIn ( " linkmotime " , tahoe_md )
2010-07-18 14:29:15 +00:00
self . failUnlessEqual ( kids [ u " bar.txt " ] [ 0 ] , " filenode " )
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( kids [ u " bar.txt " ] [ 1 ] [ " size " ] , len ( self . BAR_CONTENTS ) )
2010-07-18 14:29:15 +00:00
self . failUnlessReallyEqual ( to_str ( kids [ u " bar.txt " ] [ 1 ] [ " ro_uri " ] ) , self . _bar_txt_uri )
self . failUnlessReallyEqual ( to_str ( kids [ u " bar.txt " ] [ 1 ] [ " verify_uri " ] ) ,
2010-07-11 20:02:52 +00:00
self . _bar_txt_verifycap )
2010-06-19 02:17:18 +00:00
self . failUnlessIn ( " metadata " , kids [ u " bar.txt " ] [ 1 ] )
self . failUnlessIn ( " tahoe " , kids [ u " bar.txt " ] [ 1 ] [ " metadata " ] )
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( kids [ u " bar.txt " ] [ 1 ] [ " metadata " ] [ " tahoe " ] [ " linkcrtime " ] ,
self . _bar_txt_metadata [ " tahoe " ] [ " linkcrtime " ] )
2010-07-18 14:29:15 +00:00
self . failUnlessReallyEqual ( to_str ( kids [ u " n \u00fc .txt " ] [ 1 ] [ " ro_uri " ] ) ,
2010-07-11 20:02:52 +00:00
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-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 ] + = " -- "
2009-11-18 07:09:00 +00:00
body = " "
headers = { }
if fields :
body = " \r \n " . join ( form ) + " \r \n "
headers [ " content-type " ] = " multipart/form-data; boundary= %s " % sepbase
return self . POST2 ( urlpath , body , headers , followRedirect )
def POST2 ( self , urlpath , body = " " , headers = { } , followRedirect = False ) :
url = self . webish_url + urlpath
2007-07-08 03:06:58 +00:00
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 )
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( res . value . status , " 404 " )
2007-07-07 02:43:55 +00:00
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
2010-02-21 01:04:55 +00:00
def should302 ( self , res , which ) :
if isinstance ( res , failure . Failure ) :
res . trap ( error . Error )
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( res . value . status , " 302 " )
2010-02-21 01:04:55 +00:00
else :
self . fail ( " %s was supposed to Error(302), not get ' %s ' " %
( which , res ) )
2007-12-25 05:49:35 +00:00
2010-07-11 20:02:52 +00:00
class Web ( WebMixin , WebErrorMixin , testutil . StallMixin , testutil . ReallyEqualMixin , 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 ) :
2009-10-29 02:50:50 +00:00
self . failUnless ( ' Welcome To Tahoe-LAFS ' in res , 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 ) :
2010-06-03 23:21:05 +00:00
self . failUnless ( ' Provisioning Tool ' in res )
2007-08-28 00:26:39 +00:00
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 ) :
2010-06-03 23:21:05 +00:00
self . failUnless ( ' Provisioning Tool ' in res )
2007-08-28 00:26:39 +00:00
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
2010-01-14 22:15:29 +00:00
del _hush_pyflakes
2009-02-15 23:24:51 +00:00
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 ) :
2010-06-03 23:21:05 +00:00
self . failUnless ( ' Reliability Tool ' in res )
2009-02-15 23:24:51 +00:00
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 ) :
2010-06-03 23:21:05 +00:00
self . failUnless ( ' Reliability Tool ' in res )
2009-02-15 23:24:51 +00:00
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 ) :
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
h = self . s . get_history ( )
dl_num = h . list_all_download_statuses ( ) [ 0 ] . get_counter ( )
ul_num = h . list_all_upload_statuses ( ) [ 0 ] . get_counter ( )
mu_num = h . list_all_mapupdate_statuses ( ) [ 0 ] . get_counter ( )
pub_num = h . list_all_publish_statuses ( ) [ 0 ] . get_counter ( )
ret_num = h . 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 ) )
2010-01-14 22:15:29 +00:00
#active = data["active"]
2008-07-26 00:41:10 +00:00
# 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 )
2011-06-29 22:25:55 +00:00
d . addCallback ( lambda res : self . GET ( " /status/down- %d /event_json " % dl_num ) )
2010-08-09 22:03:42 +00:00
def _check_dl_json ( res ) :
data = simplejson . loads ( res )
self . failUnless ( isinstance ( data , dict ) )
2011-06-29 22:25:55 +00:00
self . failUnless ( " read " in data )
self . failUnlessEqual ( data [ " read " ] [ 0 ] [ " length " ] , 120 )
self . failUnlessEqual ( data [ " segment " ] [ 0 ] [ " segment_length " ] , 100 )
self . failUnlessEqual ( data [ " segment " ] [ 2 ] [ " segment_number " ] , 2 )
self . failUnlessEqual ( data [ " segment " ] [ 2 ] [ " finish_time " ] , None )
phwr_id = base32 . b2a ( hashutil . tagged_hash ( " foo " , " serverid_a " ) [ : 20 ] )
cmpu_id = base32 . b2a ( hashutil . tagged_hash ( " foo " , " serverid_b " ) [ : 20 ] )
# serverids[] keys are strings, since that's what JSON does, but
# we'd really like them to be ints
self . failUnlessEqual ( data [ " serverids " ] [ " 0 " ] , " phwr " )
2011-08-01 18:54:01 +00:00
self . failUnless ( data [ " serverids " ] . has_key ( " 1 " ) , data [ " serverids " ] )
self . failUnlessEqual ( data [ " serverids " ] [ " 1 " ] , " cmpu " , data [ " serverids " ] )
2011-06-29 22:25:55 +00:00
self . failUnlessEqual ( data [ " server_info " ] [ phwr_id ] [ " short " ] , " phwr " )
self . failUnlessEqual ( data [ " server_info " ] [ cmpu_id ] [ " short " ] , " cmpu " )
self . failUnless ( " dyhb " in data )
2010-08-09 22:03:42 +00:00
d . addCallback ( _check_dl_json )
2008-03-01 06:03:00 +00:00
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 ( )
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( drrm . render_time ( None , None ) , " " )
self . failUnlessReallyEqual ( drrm . render_time ( None , 2.5 ) , " 2.50s " )
self . failUnlessReallyEqual ( drrm . render_time ( None , 0.25 ) , " 250ms " )
self . failUnlessReallyEqual ( drrm . render_time ( None , 0.0021 ) , " 2.1ms " )
self . failUnlessReallyEqual ( drrm . render_time ( None , 0.000123 ) , " 123us " )
self . failUnlessReallyEqual ( drrm . render_rate ( None , None ) , " " )
self . failUnlessReallyEqual ( drrm . render_rate ( None , 2500000 ) , " 2.50MBps " )
self . failUnlessReallyEqual ( drrm . render_rate ( None , 30100 ) , " 30.1kBps " )
self . failUnlessReallyEqual ( drrm . render_rate ( None , 123 ) , " 123Bps " )
2008-03-04 04:56:23 +00:00
2008-03-05 21:59:56 +00:00
urrm = status . UploadResultsRendererMixin ( )
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( urrm . render_time ( None , None ) , " " )
self . failUnlessReallyEqual ( urrm . render_time ( None , 2.5 ) , " 2.50s " )
self . failUnlessReallyEqual ( urrm . render_time ( None , 0.25 ) , " 250ms " )
self . failUnlessReallyEqual ( urrm . render_time ( None , 0.0021 ) , " 2.1ms " )
self . failUnlessReallyEqual ( urrm . render_time ( None , 0.000123 ) , " 123us " )
self . failUnlessReallyEqual ( urrm . render_rate ( None , None ) , " " )
self . failUnlessReallyEqual ( urrm . render_rate ( None , 2500000 ) , " 2.50MBps " )
self . failUnlessReallyEqual ( urrm . render_rate ( None , 30100 ) , " 30.1kBps " )
self . failUnlessReallyEqual ( urrm . render_rate ( None , 123 ) , " 123Bps " )
2008-03-04 04:56:23 +00:00
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 ) ) :
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( int ( status ) , 206 )
2008-10-28 20:41:04 +00:00
self . failUnless ( headers . has_key ( " content-range " ) )
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( headers [ " content-range " ] [ 0 ] ,
" bytes 1-10/ %d " % len ( self . BAR_CONTENTS ) )
self . failUnlessReallyEqual ( res , self . BAR_CONTENTS [ 1 : 11 ] )
2008-10-28 20:41:04 +00:00
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 ) ) :
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( int ( status ) , 206 )
2008-11-18 14:41:35 +00:00
self . failUnless ( headers . has_key ( " content-range " ) )
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( headers [ " content-range " ] [ 0 ] ,
" bytes 5- %d / %d " % ( length - 1 , length ) )
self . failUnlessReallyEqual ( res , self . BAR_CONTENTS [ 5 : ] )
2008-11-18 14:41:35 +00:00
d . addCallback ( _got )
return d
2010-03-10 03:59:13 +00:00
def test_GET_FILEURL_partial_end_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 ) ) :
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( int ( status ) , 206 )
2010-03-10 03:59:13 +00:00
self . failUnless ( headers . has_key ( " content-range " ) )
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( headers [ " content-range " ] [ 0 ] ,
" bytes %d - %d / %d " % ( length - 5 , length - 1 , length ) )
self . failUnlessReallyEqual ( res , self . BAR_CONTENTS [ - 5 : ] )
2010-03-10 03:59:13 +00:00
d . addCallback ( _got )
return d
def test_GET_FILEURL_partial_range_overrun ( self ) :
headers = { " range " : " bytes=100-200 " }
d = self . shouldFail2 ( error . Error , " test_GET_FILEURL_range_overrun " ,
" 416 Requested Range not satisfiable " ,
" First beyond end of file " ,
self . GET , self . public_url + " /foo/bar.txt " ,
headers = headers )
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 ) ) :
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( res , " " )
self . failUnlessReallyEqual ( int ( status ) , 206 )
2008-10-28 20:41:04 +00:00
self . failUnless ( headers . has_key ( " content-range " ) )
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( headers [ " content-range " ] [ 0 ] ,
" bytes 1-10/ %d " % len ( self . BAR_CONTENTS ) )
2008-10-28 20:41:04 +00:00
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 ) ) :
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( int ( status ) , 206 )
2008-11-18 14:41:35 +00:00
self . failUnless ( headers . has_key ( " content-range " ) )
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( headers [ " content-range " ] [ 0 ] ,
" bytes 5- %d / %d " % ( length - 1 , length ) )
2008-11-18 14:41:35 +00:00
d . addCallback ( _got )
return d
2010-03-10 03:59:13 +00:00
def test_HEAD_FILEURL_partial_end_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 ) ) :
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( int ( status ) , 206 )
2010-03-10 03:59:13 +00:00
self . failUnless ( headers . has_key ( " content-range " ) )
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( headers [ " content-range " ] [ 0 ] ,
" bytes %d - %d / %d " % ( length - 5 , length - 1 , length ) )
2010-03-10 03:59:13 +00:00
d . addCallback ( _got )
return d
def test_HEAD_FILEURL_partial_range_overrun ( self ) :
headers = { " range " : " bytes=100-200 " }
d = self . shouldFail2 ( error . Error , " test_HEAD_FILEURL_range_overrun " ,
" 416 Requested Range not satisfiable " ,
" " ,
self . HEAD , self . public_url + " /foo/bar.txt " ,
headers = headers )
return d
2008-10-28 20:41:04 +00:00
def test_GET_FILEURL_range_bad ( self ) :
headers = { " range " : " BOGUS=fizbop-quarnak " }
2010-03-10 03:59:13 +00:00
d = self . GET ( self . public_url + " /foo/bar.txt " , headers = headers ,
return_response = True )
def _got ( ( res , status , headers ) ) :
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( int ( status ) , 200 )
2010-03-10 03:59:13 +00:00
self . failUnless ( not headers . has_key ( " content-range " ) )
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( res , self . BAR_CONTENTS )
2010-03-10 03:59:13 +00:00
d . addCallback ( _got )
2008-10-28 20:41:04 +00:00
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 ) ) :
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( res , " " )
self . failUnlessReallyEqual ( headers [ " content-length " ] [ 0 ] ,
str ( len ( self . BAR_CONTENTS ) ) )
self . failUnlessReallyEqual ( headers [ " content-type " ] , [ " text/plain " ] )
2008-05-20 18:47:43 +00:00
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 " ,
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
" 400 Bad Request " , " is not a file-cap " ,
2008-05-20 01:38:39 +00:00
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 " ,
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
" GET unknown URI type: can only do t=info " ,
2008-05-20 06:28:52 +00:00
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
2010-01-27 07:03:09 +00:00
# TODO: version of this with a Unicode filename
2007-12-12 01:04:44 +00:00
def test_GET_FILEURL_save ( self ) :
2010-01-27 07:03:09 +00:00
d = self . GET ( self . public_url + " /foo/bar.txt?filename=bar.txt&save=true " ,
return_response = True )
def _got ( ( res , statuscode , headers ) ) :
content_disposition = headers [ " content-disposition " ] [ 0 ]
self . failUnless ( content_disposition == ' attachment; filename= " bar.txt " ' , content_disposition )
self . failUnlessIsBarDotTxt ( res )
d . addCallback ( _got )
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
2009-07-20 03:46:32 +00:00
def test_PUT_overwrite_only_files ( self ) :
# create a directory, put a file in that directory.
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
contents , n , filecap = self . makefile ( 8 )
2009-07-20 03:46:32 +00:00
d = self . PUT ( self . public_url + " /foo/dir?t=mkdir " , " " )
d . addCallback ( lambda res :
self . PUT ( self . public_url + " /foo/dir/file1.txt " ,
self . NEWFILE_CONTENTS ) )
# try to overwrite the file with replace=only-files
# (this should work)
d . addCallback ( lambda res :
self . PUT ( self . public_url + " /foo/dir/file1.txt?t=uri&replace=only-files " ,
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
filecap ) )
2009-07-20 03:46:32 +00:00
d . addCallback ( lambda res :
self . shouldFail2 ( error . Error , " PUT_bad_t " , " 409 Conflict " ,
" There was already a child by that name, and you asked me "
" to not replace it " ,
self . PUT , self . public_url + " /foo/dir?t=uri&replace=only-files " ,
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
filecap ) )
2009-07-20 03:46:32 +00:00
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
2010-07-11 20:02:52 +00:00
#self.failUnlessReallyEqual(responsecode, 201)
2010-01-27 06:44:30 +00:00
d . addCallback ( self . failUnlessURIMatchesROChild , 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
2010-07-11 20:02:52 +00:00
#self.failUnlessReallyEqual(responsecode, 201)
2010-01-27 06:44:30 +00:00
d . addCallback ( self . failUnlessURIMatchesROChild , self . _foo_node , u " new.txt " )
2009-04-08 02:13:40 +00:00
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
2010-07-11 20:02:52 +00:00
#self.failUnlessReallyEqual(responsecode, 201)
2008-05-20 19:36:02 +00:00
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 )
2010-01-27 06:44:30 +00:00
d . addCallback ( self . failUnlessURIMatchesRWChild , self . _foo_node , u " new.txt " )
2008-05-20 19:36:02 +00:00
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
2010-07-11 20:02:52 +00:00
#self.failUnlessReallyEqual(responsecode, 200)
2010-01-27 06:44:30 +00:00
d . addCallback ( self . failUnlessURIMatchesROChild , 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
2010-01-27 06:44:30 +00:00
d . addCallback ( self . failUnlessURIMatchesROChild , fn , u " newdir/new.txt " )
2008-02-14 22:45:56 +00:00
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
2009-12-27 20:10:43 +00:00
def test_PUT_NEWFILEURL_emptyname ( self ) :
# an empty pathname component (i.e. a double-slash) is disallowed
d = self . shouldFail2 ( error . Error , " test_PUT_NEWFILEURL_emptyname " ,
" 400 Bad Request " ,
" The webapi does not allow empty pathname components " ,
self . PUT , self . public_url + " /foo//new.txt " , " " )
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
2009-12-27 22:54:43 +00:00
def failUnlessHasBarDotTxtMetadata ( self , res ) :
data = simplejson . loads ( res )
self . failUnless ( isinstance ( data , list ) )
2010-06-19 02:17:18 +00:00
self . failUnlessIn ( " metadata " , data [ 1 ] )
self . failUnlessIn ( " tahoe " , data [ 1 ] [ " metadata " ] )
self . failUnlessIn ( " linkcrtime " , data [ 1 ] [ " metadata " ] [ " tahoe " ] )
self . failUnlessIn ( " linkmotime " , data [ 1 ] [ " metadata " ] [ " tahoe " ] )
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( data [ 1 ] [ " metadata " ] [ " tahoe " ] [ " linkcrtime " ] ,
self . _bar_txt_metadata [ " tahoe " ] [ " linkcrtime " ] )
2009-12-27 22:54:43 +00:00
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 " )
2009-12-27 22:54:43 +00:00
def _check1 ( data ) :
self . failUnlessIsBarJSON ( data )
self . failUnlessHasBarDotTxtMetadata ( data )
return
d . addCallback ( _check1 )
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 ) :
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( res , self . _bar_txt_uri )
2007-07-07 02:43:55 +00:00
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
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( res , self . _bar_txt_uri )
2007-07-08 05:47:18 +00:00
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
2010-06-22 21:47:14 +00:00
def test_CSS_FILE ( self ) :
d = self . GET ( " /tahoe_css " , followRedirect = True )
def _check ( res ) :
CSS_STYLE = re . compile ( ' toolbar \ s { .+text-align: \ scenter.+toolbar-item.+display: \ sinline ' , re . DOTALL )
self . failUnless ( CSS_STYLE . search ( res ) , res )
d . addCallback ( _check )
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
2010-06-22 20:14:03 +00:00
def test_GET_DIRECTORY_html_banner ( self ) :
d = self . GET ( self . public_url + " /foo " , followRedirect = True )
def _check ( res ) :
self . failUnlessIn ( ' <div class= " toolbar-item " ><a href= " ../../.. " >Return to Welcome page</a></div> ' , res )
d . addCallback ( _check )
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> ' ,
2011-06-11 16:37:41 +00:00
r ' \ s+<td align= " right " > %d </td> ' % len ( self . BAR_CONTENTS ) ,
2008-06-18 02:49:40 +00:00
] )
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 )
2010-01-27 06:44:30 +00:00
# look at a readonly directory
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 + " /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 )
2010-02-25 04:18:24 +00:00
# and at a literal directory
tiny_litdir_uri = " URI:DIR2-LIT:gqytunj2onug64tufqzdcosvkjetutcjkq5gw4tvm5vwszdgnz5hgyzufqydulbshj5x2lbm " # contains one child which is itself also LIT
d . addCallback ( lambda res :
self . GET ( " /uri/ " + tiny_litdir_uri + " / " , followRedirect = True ) )
def _check5 ( res ) :
self . failUnless ( ' (immutable) ' in res , res )
self . failUnless ( re . search ( ' <td>FILE</td> '
r ' \ s+<td><a href= " [ \ . \ /]+/file/URI % 3ALIT % 3Akrugkidfnzsc4/@@named=/short " >short</a></td> ' , res ) , res )
d . addCallback ( _check5 )
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
2010-07-18 14:29:15 +00:00
self . failUnlessReallyEqual ( to_str ( got [ ( u " sub " , ) ] ) , self . _sub_uri )
2008-10-07 04:36:18 +00:00
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 ) :
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( data [ " finished " ] , True )
2008-10-22 00:03:07 +00:00
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 ( ) :
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( stats [ k ] , v ,
" stats[ %s ] was %s , not %s " %
( k , stats [ k ] , v ) )
self . failUnlessReallyEqual ( 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 " ) ]
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( len ( units ) , 7 )
2010-07-18 14:29:15 +00:00
self . failUnlessEqual ( units [ - 1 ] [ " type " ] , " stats " )
2009-01-23 05:01:36 +00:00
first = units [ 0 ]
2010-07-18 14:29:15 +00:00
self . failUnlessEqual ( first [ " path " ] , [ ] )
self . failUnlessReallyEqual ( to_str ( first [ " cap " ] ) , self . _foo_uri )
self . failUnlessEqual ( first [ " type " ] , " directory " )
baz = [ u for u in units [ : - 1 ] if to_str ( u [ " cap " ] ) == self . _baz_file_uri ] [ 0 ]
self . failUnlessEqual ( baz [ " path " ] , [ " sub " , " baz.txt " ] )
2009-01-23 05:01:36 +00:00
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 ) :
2010-07-18 14:29:15 +00:00
self . failUnlessReallyEqual ( to_str ( res ) , self . _foo_uri )
2007-07-07 02:43:55 +00:00
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 ) :
2010-07-18 14:29:15 +00:00
self . failUnlessReallyEqual ( to_str ( res ) , self . _foo_readonly_uri )
2007-07-07 02:43:55 +00:00
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
2009-12-27 20:10:43 +00:00
def test_POST_NEWDIRURL ( self ) :
d = self . POST2 ( 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_NEWDIRURL_emptyname ( self ) :
# an empty pathname component (i.e. a double-slash) is disallowed
d = self . shouldFail2 ( error . Error , " test_POST_NEWDIRURL_emptyname " ,
" 400 Bad Request " ,
" The webapi does not allow empty pathname components, i.e. a double slash " ,
self . POST , self . public_url + " //?t=mkdir " )
return d
2009-10-13 02:34:44 +00:00
def test_POST_NEWDIRURL_initial_children ( self ) :
2010-01-27 06:44:30 +00:00
( newkids , caps ) = self . _create_initial_children ( )
2009-11-18 07:09:00 +00:00
d = self . POST2 ( self . public_url + " /foo/newdir?t=mkdir-with-children " ,
simplejson . dumps ( newkids ) )
2009-10-13 02:34:44 +00:00
def _check ( uri ) :
n = self . s . create_node_from_uri ( uri . strip ( ) )
d2 = self . failUnlessNodeKeysAre ( n , newkids . keys ( ) )
d2 . addCallback ( lambda ign :
2010-01-27 06:44:30 +00:00
self . failUnlessROChildURIIs ( n , u " child-imm " ,
caps [ ' filecap1 ' ] ) )
2009-10-13 02:34:44 +00:00
d2 . addCallback ( lambda ign :
2010-01-27 06:44:30 +00:00
self . failUnlessRWChildURIIs ( n , u " child-mutable " ,
caps [ ' filecap2 ' ] ) )
2009-10-13 02:34:44 +00:00
d2 . addCallback ( lambda ign :
2010-01-27 06:44:30 +00:00
self . failUnlessROChildURIIs ( n , u " child-mutable-ro " ,
caps [ ' filecap3 ' ] ) )
2009-10-13 02:34:44 +00:00
d2 . addCallback ( lambda ign :
2010-01-27 06:44:30 +00:00
self . failUnlessROChildURIIs ( n , u " unknownchild-ro " ,
caps [ ' unknown_rocap ' ] ) )
d2 . addCallback ( lambda ign :
self . failUnlessRWChildURIIs ( n , u " unknownchild-rw " ,
caps [ ' unknown_rwcap ' ] ) )
d2 . addCallback ( lambda ign :
self . failUnlessROChildURIIs ( n , u " unknownchild-imm " ,
caps [ ' unknown_immcap ' ] ) )
d2 . addCallback ( lambda ign :
self . failUnlessRWChildURIIs ( n , u " dirchild " ,
caps [ ' dircap ' ] ) )
2010-02-22 02:53:52 +00:00
d2 . addCallback ( lambda ign :
self . failUnlessROChildURIIs ( n , u " dirchild-lit " ,
caps [ ' litdircap ' ] ) )
d2 . addCallback ( lambda ign :
self . failUnlessROChildURIIs ( n , u " dirchild-empty " ,
caps [ ' emptydircap ' ] ) )
2009-10-13 02:34:44 +00:00
return d2
d . addCallback ( _check )
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 , newkids . keys ( ) )
d . addCallback ( lambda res : self . _foo_node . get ( u " newdir " ) )
2010-01-27 06:44:30 +00:00
d . addCallback ( self . failUnlessROChildURIIs , u " child-imm " , caps [ ' filecap1 ' ] )
2009-10-13 02:34:44 +00:00
return d
2009-11-18 07:09:00 +00:00
def test_POST_NEWDIRURL_immutable ( self ) :
2010-01-27 06:44:30 +00:00
( newkids , caps ) = self . _create_immutable_children ( )
2009-11-18 07:09:00 +00:00
d = self . POST2 ( self . public_url + " /foo/newdir?t=mkdir-immutable " ,
simplejson . dumps ( newkids ) )
def _check ( uri ) :
n = self . s . create_node_from_uri ( uri . strip ( ) )
d2 = self . failUnlessNodeKeysAre ( n , newkids . keys ( ) )
d2 . addCallback ( lambda ign :
2010-01-27 06:44:30 +00:00
self . failUnlessROChildURIIs ( n , u " child-imm " ,
caps [ ' filecap1 ' ] ) )
d2 . addCallback ( lambda ign :
self . failUnlessROChildURIIs ( n , u " unknownchild-imm " ,
caps [ ' unknown_immcap ' ] ) )
2009-11-18 07:09:00 +00:00
d2 . addCallback ( lambda ign :
2010-01-27 06:44:30 +00:00
self . failUnlessROChildURIIs ( n , u " dirchild-imm " ,
caps [ ' immdircap ' ] ) )
2010-02-22 02:53:52 +00:00
d2 . addCallback ( lambda ign :
self . failUnlessROChildURIIs ( n , u " dirchild-lit " ,
caps [ ' litdircap ' ] ) )
d2 . addCallback ( lambda ign :
self . failUnlessROChildURIIs ( n , u " dirchild-empty " ,
caps [ ' emptydircap ' ] ) )
2009-11-18 07:09:00 +00:00
return d2
d . addCallback ( _check )
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 , newkids . keys ( ) )
d . addCallback ( lambda res : self . _foo_node . get ( u " newdir " ) )
2010-01-27 06:44:30 +00:00
d . addCallback ( self . failUnlessROChildURIIs , u " child-imm " , caps [ ' filecap1 ' ] )
2009-11-18 07:09:00 +00:00
d . addCallback ( lambda res : self . _foo_node . get ( u " newdir " ) )
2010-01-27 06:44:30 +00:00
d . addCallback ( self . failUnlessROChildURIIs , u " unknownchild-imm " , caps [ ' unknown_immcap ' ] )
d . addCallback ( lambda res : self . _foo_node . get ( u " newdir " ) )
d . addCallback ( self . failUnlessROChildURIIs , u " dirchild-imm " , caps [ ' immdircap ' ] )
2010-02-22 02:53:52 +00:00
d . addCallback ( lambda res : self . _foo_node . get ( u " newdir " ) )
d . addCallback ( self . failUnlessROChildURIIs , u " dirchild-lit " , caps [ ' litdircap ' ] )
d . addCallback ( lambda res : self . _foo_node . get ( u " newdir " ) )
d . addCallback ( self . failUnlessROChildURIIs , u " dirchild-empty " , caps [ ' emptydircap ' ] )
2009-11-18 07:09:00 +00:00
d . addErrback ( self . explain_web_error )
return d
def test_POST_NEWDIRURL_immutable_bad ( self ) :
2010-01-27 06:44:30 +00:00
( newkids , caps ) = self . _create_initial_children ( )
2009-11-18 07:09:00 +00:00
d = self . shouldFail2 ( error . Error , " test_POST_NEWDIRURL_immutable_bad " ,
" 400 Bad Request " ,
2010-01-27 06:44:30 +00:00
" needed to be immutable but was not " ,
2009-11-18 07:09:00 +00:00
self . POST2 ,
self . public_url + " /foo/newdir?t=mkdir-immutable " ,
simplejson . dumps ( newkids ) )
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 " )
2010-07-11 20:02:52 +00:00
d . addCallback ( lambda ssnode : self . failUnlessReallyEqual ( ssnode . get_uri ( ) , ssuri ) )
2008-03-18 20:11:08 +00:00
d = self . POST ( url )
2010-07-11 20:02:52 +00:00
d . addCallback ( lambda uri2 : self . failUnlessReallyEqual ( uri2 , ssuri ) )
2008-03-18 20:11:08 +00:00
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 ) :
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( sorted ( children . keys ( ) ) , sorted ( expected_keys ) )
2007-12-05 06:01:37 +00:00
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 )
2009-12-01 22:44:35 +00:00
d . addCallback ( lambda node : download_to_data ( node ) )
2007-12-05 06:01:37 +00:00
def _check ( contents ) :
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( contents , expected_contents )
2007-12-05 06:01:37 +00:00
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 ) :
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( contents , expected_contents )
2008-04-18 00:51:38 +00:00
d . addCallback ( _check )
return d
2010-01-27 06:44:30 +00:00
def failUnlessRWChildURIIs ( self , node , name , expected_uri ) :
assert isinstance ( name , unicode )
d = node . get_child_at_path ( name )
def _check ( child ) :
self . failUnless ( child . is_unknown ( ) or not child . is_readonly ( ) )
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( child . get_uri ( ) , expected_uri . strip ( ) )
self . failUnlessReallyEqual ( child . get_write_uri ( ) , expected_uri . strip ( ) )
2010-01-27 06:44:30 +00:00
expected_ro_uri = self . _make_readonly ( expected_uri )
if expected_ro_uri :
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( child . get_readonly_uri ( ) , expected_ro_uri . strip ( ) )
2010-01-27 06:44:30 +00:00
d . addCallback ( _check )
return d
def failUnlessROChildURIIs ( 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 ) :
2010-01-27 06:44:30 +00:00
self . failUnless ( child . is_unknown ( ) or child . is_readonly ( ) )
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( child . get_write_uri ( ) , None )
self . failUnlessReallyEqual ( child . get_uri ( ) , expected_uri . strip ( ) )
self . failUnlessReallyEqual ( child . get_readonly_uri ( ) , expected_uri . strip ( ) )
2007-12-05 06:01:37 +00:00
d . addCallback ( _check )
return d
2010-01-27 06:44:30 +00:00
def failUnlessURIMatchesRWChild ( 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 ) :
2010-01-27 06:44:30 +00:00
self . failUnless ( child . is_unknown ( ) or not child . is_readonly ( ) )
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( child . get_uri ( ) , got_uri . strip ( ) )
self . failUnlessReallyEqual ( child . get_write_uri ( ) , got_uri . strip ( ) )
2010-01-27 06:44:30 +00:00
expected_ro_uri = self . _make_readonly ( got_uri )
if expected_ro_uri :
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( child . get_readonly_uri ( ) , expected_ro_uri . strip ( ) )
2010-01-27 06:44:30 +00:00
d . addCallback ( _check )
return d
def failUnlessURIMatchesROChild ( self , got_uri , node , name ) :
assert isinstance ( name , unicode )
d = node . get_child_at_path ( name )
def _check ( child ) :
self . failUnless ( child . is_unknown ( ) or child . is_readonly ( ) )
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( child . get_write_uri ( ) , None )
self . failUnlessReallyEqual ( got_uri . strip ( ) , child . get_uri ( ) )
self . failUnlessReallyEqual ( got_uri . strip ( ) , child . get_readonly_uri ( ) )
2007-12-05 06:01:37 +00:00
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
2010-01-27 06:44:30 +00:00
d . addCallback ( self . failUnlessURIMatchesROChild , fn , u " new.txt " )
2008-02-14 22:45:56 +00:00
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
2010-01-27 06:44:30 +00:00
d . addCallback ( self . failUnlessURIMatchesROChild , 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 ) )
2010-07-11 20:02:52 +00:00
d . addCallback ( lambda contents : self . failUnlessReallyEqual ( contents ,
self . NEWFILE_CONTENTS ,
contents ) )
2008-06-04 00:09:39 +00:00
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
2010-01-27 06:44:30 +00:00
d . addCallback ( self . failUnlessURIMatchesROChild , fn , filename )
2008-06-04 00:09:39 +00:00
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 ) )
2010-07-11 20:02:52 +00:00
d . addCallback ( lambda contents : self . failUnlessReallyEqual ( 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 ) :
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( statuscode , str ( http . FOUND ) )
2008-05-20 06:28:52 +00:00
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 :
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( res , self . NEWFILE_CONTENTS ) )
2008-05-20 06:28:52 +00:00
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 ) )
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
def _check ( filecap ) :
filecap = filecap . strip ( )
self . failUnless ( filecap . startswith ( " URI:SSK: " ) , filecap )
self . filecap = filecap
u = uri . WriteableSSKFileURI . init_from_string ( filecap )
2010-02-22 02:45:04 +00:00
self . failUnless ( u . get_storage_index ( ) in FakeMutableFileNode . all_contents )
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
n = self . s . create_node_from_uri ( filecap )
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 ) :
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( data , self . NEWFILE_CONTENTS )
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
return self . GET ( " /uri/ %s " % urllib . quote ( self . filecap ) )
2008-02-06 05:10:22 +00:00
d . addCallback ( _check2 )
2008-05-20 01:38:39 +00:00
def _check3 ( data ) :
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( data , self . NEWFILE_CONTENTS )
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
return self . GET ( " /file/ %s " % urllib . quote ( self . filecap ) )
2008-05-20 01:38:39 +00:00
d . addCallback ( _check3 )
def _check4 ( data ) :
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( data , self . NEWFILE_CONTENTS )
2008-05-20 01:38:39 +00:00
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
2010-01-27 06:44:30 +00:00
d . addCallback ( self . failUnlessURIMatchesRWChild , 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 ) ) )
2010-01-27 06:44:30 +00:00
d . addCallback ( self . failUnlessURIMatchesRWChild , 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 ( ) )
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( self . _mutable_uri , newnode . get_uri ( ) )
2007-12-05 06:42:54 +00:00
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 ) )
2010-01-27 06:44:30 +00:00
d . addCallback ( self . failUnlessURIMatchesRWChild , fn , u " new.txt " )
2008-05-20 01:38:39 +00:00
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 ( ) )
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( self . _mutable_uri , newnode . get_uri ( ) )
2007-12-15 00:52:05 +00:00
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 )
2010-07-18 14:29:15 +00:00
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 ( ) ] )
2010-07-18 14:29:15 +00:00
self . failUnless ( u " new.txt " in children )
new_json = children [ u " new.txt " ]
self . failUnlessEqual ( new_json [ 0 ] , " filenode " )
2008-05-20 22:40:49 +00:00
self . failUnless ( new_json [ 1 ] [ " mutable " ] )
2010-07-18 14:29:15 +00:00
self . failUnlessReallyEqual ( to_str ( new_json [ 1 ] [ " rw_uri " ] ) , self . _mutable_uri )
2010-07-11 20:02:52 +00:00
ro_uri = self . _mutable_node . get_readonly ( ) . to_string ( )
2010-07-18 14:29:15 +00:00
self . failUnlessReallyEqual ( to_str ( new_json [ 1 ] [ " ro_uri " ] ) , ro_uri )
2008-05-20 01:37:28 +00:00
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 )
2010-07-18 14:29:15 +00:00
self . failUnlessEqual ( parsed [ 0 ] , " filenode " )
2008-05-20 22:40:49 +00:00
self . failUnless ( parsed [ 1 ] [ " mutable " ] )
2010-07-18 14:29:15 +00:00
self . failUnlessReallyEqual ( to_str ( parsed [ 1 ] [ " rw_uri " ] ) , self . _mutable_uri )
2010-07-11 20:02:52 +00:00
ro_uri = self . _mutable_node . get_readonly ( ) . to_string ( )
2010-07-18 14:29:15 +00:00
self . failUnlessReallyEqual ( to_str ( parsed [ 1 ] [ " ro_uri " ] ) , ro_uri )
2008-05-20 22:14:19 +00:00
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 " ) )
2010-07-11 20:02:52 +00:00
d . addCallback ( lambda res : self . failUnlessReallyEqual ( res , self . _mutable_uri ) )
2008-05-20 18:33:14 +00:00
d . addCallback ( lambda res :
self . GET ( self . public_url + " /foo/new.txt?t=readonly-uri " ) )
def _check_ro_uri ( res ) :
2010-07-11 20:02:52 +00:00
ro_uri = self . _mutable_node . get_readonly ( ) . to_string ( )
self . failUnlessReallyEqual ( res , ro_uri )
2008-05-20 18:33:14 +00:00
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 :
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( 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 ) ) :
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( res , " " )
self . failUnlessReallyEqual ( headers [ " content-length " ] [ 0 ] ,
str ( len ( NEW2_CONTENTS ) ) )
self . failUnlessReallyEqual ( headers [ " content-type " ] , [ " text/plain " ] )
2008-05-20 18:47:43 +00:00
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 ,
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
" test_POST_upload_mutable_toobig " ,
2008-06-03 07:03:16 +00:00
" 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
2010-01-27 06:44:30 +00:00
d . addCallback ( self . failUnlessURIMatchesROChild , 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 " ) )
2010-07-11 20:02:52 +00:00
d . addCallback ( lambda res : self . failUnlessReallyEqual ( res ,
self . NEWFILE_CONTENTS ) )
2007-12-25 10:48:57 +00:00
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 )
2010-01-27 06:44:30 +00:00
d . addCallback ( self . failUnlessURIMatchesROChild , 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 ) :
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( statuscode , str ( http . FOUND ) )
self . failUnlessReallyEqual ( target , redir_url )
2008-07-16 22:42:56 +00:00
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 ) :
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( statuscode , str ( http . FOUND ) )
self . failUnlessReallyEqual ( target , redir_url )
2008-09-07 19:44:56 +00:00
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 ) :
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( statuscode , str ( http . FOUND ) )
self . failUnlessReallyEqual ( target , redir_url )
2008-07-16 22:42:56 +00:00
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 ) :
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( statuscode , str ( http . FOUND ) )
self . failUnlessReallyEqual ( target , redir_url )
2008-09-07 19:44:56 +00:00
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 ) :
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( 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 ) :
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( data [ " finished " ] , True )
self . failUnlessReallyEqual ( data [ " count-objects-checked " ] , 8 )
self . failUnlessReallyEqual ( data [ " count-objects-healthy " ] , 8 )
2008-10-22 00:03:07 +00:00
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 )
2010-07-18 14:29:15 +00:00
self . failUnlessEqual ( data [ " storage-index " ] , foo_si_s )
2008-10-24 00:11:18 +00:00
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 ) :
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( data [ " finished " ] , True )
self . failUnlessReallyEqual ( data [ " count-objects-checked " ] , 8 )
self . failUnlessReallyEqual ( data [ " count-objects-healthy-pre-repair " ] , 8 )
self . failUnlessReallyEqual ( data [ " count-objects-unhealthy-pre-repair " ] , 0 )
self . failUnlessReallyEqual ( data [ " count-corrupt-shares-pre-repair " ] , 0 )
self . failUnlessReallyEqual ( data [ " count-repairs-attempted " ] , 0 )
self . failUnlessReallyEqual ( data [ " count-repairs-successful " ] , 0 )
self . failUnlessReallyEqual ( data [ " count-repairs-unsuccessful " ] , 0 )
self . failUnlessReallyEqual ( data [ " count-objects-healthy-post-repair " ] , 8 )
self . failUnlessReallyEqual ( data [ " count-objects-unhealthy-post-repair " ] , 0 )
self . failUnlessReallyEqual ( data [ " count-corrupt-shares-post-repair " ] , 0 )
2008-10-22 00:03:07 +00:00
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
2009-10-13 02:34:44 +00:00
def test_POST_mkdir_initial_children ( self ) :
2010-01-27 06:44:30 +00:00
( newkids , caps ) = self . _create_initial_children ( )
2009-11-18 07:09:00 +00:00
d = self . POST2 ( self . public_url +
" /foo?t=mkdir-with-children&name=newdir " ,
simplejson . dumps ( newkids ) )
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 , newkids . keys ( ) )
d . addCallback ( lambda res : self . _foo_node . get ( u " newdir " ) )
2010-01-27 06:44:30 +00:00
d . addCallback ( self . failUnlessROChildURIIs , u " child-imm " , caps [ ' filecap1 ' ] )
2009-11-18 07:09:00 +00:00
return d
def test_POST_mkdir_immutable ( self ) :
2010-01-27 06:44:30 +00:00
( newkids , caps ) = self . _create_immutable_children ( )
2009-11-18 07:09:00 +00:00
d = self . POST2 ( self . public_url +
" /foo?t=mkdir-immutable&name=newdir " ,
simplejson . dumps ( newkids ) )
2009-10-13 02:34:44 +00:00
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 , newkids . keys ( ) )
d . addCallback ( lambda res : self . _foo_node . get ( u " newdir " ) )
2010-01-27 06:44:30 +00:00
d . addCallback ( self . failUnlessROChildURIIs , u " child-imm " , caps [ ' filecap1 ' ] )
2009-11-18 07:09:00 +00:00
d . addCallback ( lambda res : self . _foo_node . get ( u " newdir " ) )
2010-01-27 06:44:30 +00:00
d . addCallback ( self . failUnlessROChildURIIs , u " unknownchild-imm " , caps [ ' unknown_immcap ' ] )
d . addCallback ( lambda res : self . _foo_node . get ( u " newdir " ) )
d . addCallback ( self . failUnlessROChildURIIs , u " dirchild-imm " , caps [ ' immdircap ' ] )
2010-02-22 02:53:52 +00:00
d . addCallback ( lambda res : self . _foo_node . get ( u " newdir " ) )
d . addCallback ( self . failUnlessROChildURIIs , u " dirchild-lit " , caps [ ' litdircap ' ] )
d . addCallback ( lambda res : self . _foo_node . get ( u " newdir " ) )
d . addCallback ( self . failUnlessROChildURIIs , u " dirchild-empty " , caps [ ' emptydircap ' ] )
2009-11-18 07:09:00 +00:00
return d
def test_POST_mkdir_immutable_bad ( self ) :
2010-01-27 06:44:30 +00:00
( newkids , caps ) = self . _create_initial_children ( )
2009-11-18 07:09:00 +00:00
d = self . shouldFail2 ( error . Error , " test_POST_mkdir_immutable_bad " ,
" 400 Bad Request " ,
2010-01-27 06:44:30 +00:00
" needed to be immutable but was not " ,
2009-11-18 07:09:00 +00:00
self . POST2 ,
self . public_url +
" /foo?t=mkdir-immutable&name=newdir " ,
simplejson . dumps ( newkids ) )
2009-10-13 02:34:44 +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 ) :
2009-07-17 01:01:03 +00:00
uri . DirectoryURI . init_from_string ( res )
2007-12-18 20:15:08 +00:00
d . addCallback ( _after_mkdir )
return d
2010-01-21 06:50:52 +00:00
def test_POST_mkdir_no_parentdir_noredirect2 ( self ) :
# make sure form-based arguments (as on the welcome page) still work
d = self . POST ( " /uri " , t = " mkdir " )
def _after_mkdir ( res ) :
uri . DirectoryURI . init_from_string ( res )
d . addCallback ( _after_mkdir )
d . addErrback ( self . explain_web_error )
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
2010-01-21 06:50:52 +00:00
def test_POST_mkdir_no_parentdir_redirect2 ( self ) :
d = self . POST ( " /uri " , t = " mkdir " , redirect_to_result = " true " )
d . addBoth ( self . shouldRedirect , None , statuscode = ' 303 ' )
def _check_target ( target ) :
target = urllib . unquote ( target )
self . failUnless ( target . startswith ( " uri/URI:DIR2: " ) , target )
d . addCallback ( _check_target )
d . addErrback ( self . explain_web_error )
return d
2010-01-27 06:44:30 +00:00
def _make_readonly ( self , u ) :
ro_uri = uri . from_string ( u ) . get_readonly ( )
if ro_uri is None :
return None
return ro_uri . to_string ( )
2009-10-13 02:34:44 +00:00
def _create_initial_children ( self ) :
contents , n , filecap1 = self . makefile ( 12 )
md1 = { " metakey1 " : " metavalue1 " }
filecap2 = make_mutable_file_uri ( )
node3 = self . s . create_node_from_uri ( make_mutable_file_uri ( ) )
filecap3 = node3 . get_readonly_uri ( )
node4 = self . s . create_node_from_uri ( make_mutable_file_uri ( ) )
dircap = DirectoryNode ( node4 , None , None ) . get_uri ( )
2010-02-22 02:53:52 +00:00
litdircap = " URI:DIR2-LIT:ge3dumj2mewdcotyfqydulbshj5x2lbm "
emptydircap = " URI:DIR2-LIT: "
2010-01-27 06:44:30 +00:00
newkids = { u " child-imm " : [ " filenode " , { " rw_uri " : filecap1 ,
" ro_uri " : self . _make_readonly ( filecap1 ) ,
" metadata " : md1 , } ] ,
u " child-mutable " : [ " filenode " , { " rw_uri " : filecap2 ,
" ro_uri " : self . _make_readonly ( filecap2 ) } ] ,
2009-10-13 02:34:44 +00:00
u " child-mutable-ro " : [ " filenode " , { " ro_uri " : filecap3 } ] ,
2010-01-27 06:44:30 +00:00
u " unknownchild-rw " : [ " unknown " , { " rw_uri " : unknown_rwcap ,
" ro_uri " : unknown_rocap } ] ,
u " unknownchild-ro " : [ " unknown " , { " ro_uri " : unknown_rocap } ] ,
u " unknownchild-imm " : [ " unknown " , { " ro_uri " : unknown_immcap } ] ,
u " dirchild " : [ " dirnode " , { " rw_uri " : dircap ,
" ro_uri " : self . _make_readonly ( dircap ) } ] ,
2010-02-22 02:53:52 +00:00
u " dirchild-lit " : [ " dirnode " , { " ro_uri " : litdircap } ] ,
u " dirchild-empty " : [ " dirnode " , { " ro_uri " : emptydircap } ] ,
2009-10-13 02:34:44 +00:00
}
2010-01-27 06:44:30 +00:00
return newkids , { ' filecap1 ' : filecap1 ,
' filecap2 ' : filecap2 ,
' filecap3 ' : filecap3 ,
' unknown_rwcap ' : unknown_rwcap ,
' unknown_rocap ' : unknown_rocap ,
' unknown_immcap ' : unknown_immcap ,
2010-02-22 02:53:52 +00:00
' dircap ' : dircap ,
' litdircap ' : litdircap ,
' emptydircap ' : emptydircap }
2009-10-13 02:34:44 +00:00
2009-11-18 07:09:00 +00:00
def _create_immutable_children ( self ) :
contents , n , filecap1 = self . makefile ( 12 )
md1 = { " metakey1 " : " metavalue1 " }
tnode = create_chk_filenode ( " immutable directory contents \n " * 10 )
dnode = DirectoryNode ( tnode , None , None )
assert not dnode . is_mutable ( )
immdircap = dnode . get_uri ( )
2010-02-22 02:53:52 +00:00
litdircap = " URI:DIR2-LIT:ge3dumj2mewdcotyfqydulbshj5x2lbm "
emptydircap = " URI:DIR2-LIT: "
2010-01-27 06:44:30 +00:00
newkids = { u " child-imm " : [ " filenode " , { " ro_uri " : filecap1 ,
" metadata " : md1 , } ] ,
u " unknownchild-imm " : [ " unknown " , { " ro_uri " : unknown_immcap } ] ,
u " dirchild-imm " : [ " dirnode " , { " ro_uri " : immdircap } ] ,
2010-02-22 02:53:52 +00:00
u " dirchild-lit " : [ " dirnode " , { " ro_uri " : litdircap } ] ,
u " dirchild-empty " : [ " dirnode " , { " ro_uri " : emptydircap } ] ,
2009-11-18 07:09:00 +00:00
}
2010-01-27 06:44:30 +00:00
return newkids , { ' filecap1 ' : filecap1 ,
' unknown_immcap ' : unknown_immcap ,
2010-02-22 02:53:52 +00:00
' immdircap ' : immdircap ,
' litdircap ' : litdircap ,
' emptydircap ' : emptydircap }
2009-11-18 07:09:00 +00:00
2009-10-13 02:34:44 +00:00
def test_POST_mkdir_no_parentdir_initial_children ( self ) :
2010-01-27 06:44:30 +00:00
( newkids , caps ) = self . _create_initial_children ( )
2009-11-18 07:09:00 +00:00
d = self . POST2 ( " /uri?t=mkdir-with-children " , simplejson . dumps ( newkids ) )
2009-10-13 02:34:44 +00:00
def _after_mkdir ( res ) :
self . failUnless ( res . startswith ( " URI:DIR " ) , res )
n = self . s . create_node_from_uri ( res )
d2 = self . failUnlessNodeKeysAre ( n , newkids . keys ( ) )
d2 . addCallback ( lambda ign :
2010-01-27 06:44:30 +00:00
self . failUnlessROChildURIIs ( n , u " child-imm " ,
caps [ ' filecap1 ' ] ) )
2009-10-13 02:34:44 +00:00
d2 . addCallback ( lambda ign :
2010-01-27 06:44:30 +00:00
self . failUnlessRWChildURIIs ( n , u " child-mutable " ,
caps [ ' filecap2 ' ] ) )
2009-10-13 02:34:44 +00:00
d2 . addCallback ( lambda ign :
2010-01-27 06:44:30 +00:00
self . failUnlessROChildURIIs ( n , u " child-mutable-ro " ,
caps [ ' filecap3 ' ] ) )
2009-10-13 02:34:44 +00:00
d2 . addCallback ( lambda ign :
2010-01-27 06:44:30 +00:00
self . failUnlessRWChildURIIs ( n , u " unknownchild-rw " ,
caps [ ' unknown_rwcap ' ] ) )
d2 . addCallback ( lambda ign :
self . failUnlessROChildURIIs ( n , u " unknownchild-ro " ,
caps [ ' unknown_rocap ' ] ) )
d2 . addCallback ( lambda ign :
self . failUnlessROChildURIIs ( n , u " unknownchild-imm " ,
caps [ ' unknown_immcap ' ] ) )
d2 . addCallback ( lambda ign :
self . failUnlessRWChildURIIs ( n , u " dirchild " ,
caps [ ' dircap ' ] ) )
2009-10-13 02:34:44 +00:00
return d2
d . addCallback ( _after_mkdir )
return d
2009-10-26 01:13:21 +00:00
def test_POST_mkdir_no_parentdir_unexpected_children ( self ) :
# the regular /uri?t=mkdir operation is specified to ignore its body.
# Only t=mkdir-with-children pays attention to it.
2010-01-27 06:44:30 +00:00
( newkids , caps ) = self . _create_initial_children ( )
2009-10-26 01:13:21 +00:00
d = self . shouldHTTPError ( " POST t=mkdir unexpected children " ,
400 , " Bad Request " ,
" t=mkdir does not accept children=, "
" try t=mkdir-with-children instead " ,
2009-11-18 07:09:00 +00:00
self . POST2 , " /uri?t=mkdir " , # without children
simplejson . dumps ( newkids ) )
2009-10-26 01:13:21 +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
2009-11-18 07:09:00 +00:00
def test_POST_mkdir_no_parentdir_immutable ( self ) :
2010-01-27 06:44:30 +00:00
( newkids , caps ) = self . _create_immutable_children ( )
2009-11-18 07:09:00 +00:00
d = self . POST2 ( " /uri?t=mkdir-immutable " , simplejson . dumps ( newkids ) )
def _after_mkdir ( res ) :
self . failUnless ( res . startswith ( " URI:DIR " ) , res )
n = self . s . create_node_from_uri ( res )
d2 = self . failUnlessNodeKeysAre ( n , newkids . keys ( ) )
d2 . addCallback ( lambda ign :
2010-01-27 06:44:30 +00:00
self . failUnlessROChildURIIs ( n , u " child-imm " ,
caps [ ' filecap1 ' ] ) )
d2 . addCallback ( lambda ign :
self . failUnlessROChildURIIs ( n , u " unknownchild-imm " ,
caps [ ' unknown_immcap ' ] ) )
2009-11-18 07:09:00 +00:00
d2 . addCallback ( lambda ign :
2010-01-27 06:44:30 +00:00
self . failUnlessROChildURIIs ( n , u " dirchild-imm " ,
caps [ ' immdircap ' ] ) )
2010-02-22 02:53:52 +00:00
d2 . addCallback ( lambda ign :
self . failUnlessROChildURIIs ( n , u " dirchild-lit " ,
caps [ ' litdircap ' ] ) )
d2 . addCallback ( lambda ign :
self . failUnlessROChildURIIs ( n , u " dirchild-empty " ,
caps [ ' emptydircap ' ] ) )
2009-11-18 07:09:00 +00:00
return d2
d . addCallback ( _after_mkdir )
return d
def test_POST_mkdir_no_parentdir_immutable_bad ( self ) :
2010-01-27 06:44:30 +00:00
( newkids , caps ) = self . _create_initial_children ( )
2009-11-18 07:09:00 +00:00
d = self . shouldFail2 ( error . Error ,
" test_POST_mkdir_no_parentdir_immutable_bad " ,
" 400 Bad Request " ,
2010-01-27 06:44:30 +00:00
" needed to be immutable but was not " ,
2009-11-18 07:09:00 +00:00
self . POST2 ,
" /uri?t=mkdir-immutable " ,
simplejson . dumps ( newkids ) )
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 ) :
2010-01-27 07:03:09 +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
2010-01-24 03:00:20 +00:00
def test_POST_set_children ( self , command_name = " set_children " ) :
2008-03-01 01:40:27 +00:00
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)
2010-01-24 03:00:20 +00:00
url = self . webish_url + self . public_url + " /foo " + " ?t= " + command_name
2008-03-01 01:40:27 +00:00
d = client . getPage ( url , method = " POST " , postdata = reqbody )
def _then ( res ) :
2010-01-27 06:44:30 +00:00
self . failUnlessURIMatchesROChild ( newuri9 , self . _foo_node , u " atomic_added_1 " )
self . failUnlessURIMatchesROChild ( newuri10 , self . _foo_node , u " atomic_added_2 " )
self . failUnlessURIMatchesROChild ( newuri11 , self . _foo_node , u " atomic_added_3 " )
2008-03-01 01:40:27 +00:00
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
2010-01-24 03:00:20 +00:00
def test_POST_set_children_with_hyphen ( self ) :
return self . test_POST_set_children ( command_name = " set-children " )
2010-01-27 23:06:42 +00:00
def test_POST_link_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 )
2010-01-27 06:44:30 +00:00
d . addCallback ( self . failUnlessURIMatchesROChild , 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
2010-01-27 23:06:42 +00:00
def test_POST_link_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 )
2010-01-27 06:44:30 +00:00
d . addCallback ( self . failUnlessURIMatchesROChild , 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
2010-01-27 23:06:42 +00:00
def test_POST_link_uri_unknown_bad ( self ) :
2010-05-19 05:51:46 +00:00
d = self . POST ( self . public_url + " /foo " , t = " uri " , name = " future.txt " , uri = unknown_rwcap )
2010-01-27 23:06:42 +00:00
d . addBoth ( self . shouldFail , error . Error ,
" POST_link_uri_unknown_bad " ,
" 400 Bad Request " ,
" unknown cap in a write slot " )
return d
def test_POST_link_uri_unknown_ro_good ( self ) :
2010-05-19 05:51:46 +00:00
d = self . POST ( self . public_url + " /foo " , t = " uri " , name = " future-ro.txt " , uri = unknown_rocap )
2010-01-27 23:06:42 +00:00
d . addCallback ( self . failUnlessURIMatchesROChild , self . _foo_node , u " future-ro.txt " )
return d
def test_POST_link_uri_unknown_imm_good ( self ) :
2010-05-19 05:51:46 +00:00
d = self . POST ( self . public_url + " /foo " , t = " uri " , name = " future-imm.txt " , uri = unknown_immcap )
2010-01-27 23:06:42 +00:00
d . addCallback ( self . failUnlessURIMatchesROChild , self . _foo_node , u " future-imm.txt " )
return d
def test_POST_link_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 ,
2010-01-27 23:06:42 +00:00
" POST_link_uri_no_replace_queryarg " ,
2007-08-15 22:21:38 +00:00
" 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
2010-01-27 23:06:42 +00:00
def test_POST_link_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 ,
2010-01-27 23:06:42 +00:00
" POST_link_uri_no_replace_field " ,
2007-08-15 22:21:38 +00:00
" 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 )
2010-07-18 14:29:15 +00:00
self . failUnlessEqual ( data [ 0 ] , " dirnode " , data )
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( len ( data [ 1 ] [ " children " ] ) , 0 )
2007-08-15 22:21:38 +00:00
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 :
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( 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
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( 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 ) :
2009-10-12 22:45:06 +00:00
d = self . s . create_dirnode ( )
2008-10-29 04:54:46 +00:00
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 :
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( res . strip ( ) , new_uri ) )
2008-10-29 04:54:46 +00:00
d . addCallback ( lambda res :
2010-01-27 06:44:30 +00:00
self . failUnlessRWChildURIIs ( self . public_root ,
u " foo " ,
new_uri ) )
2008-10-29 04:54:46 +00:00
return d
d . addCallback ( _made_dir )
return d
2008-10-29 04:57:44 +00:00
def test_PUT_DIRURL_uri_noreplace ( self ) :
2009-10-12 22:45:06 +00:00
d = self . s . create_dirnode ( )
2008-10-29 04:57:44 +00:00
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 :
2010-01-27 06:44:30 +00:00
self . failUnlessRWChildURIIs ( self . public_root ,
u " foo " ,
self . _foo_uri ) )
2008-10-29 04:57:44 +00:00
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 :
2010-01-27 06:44:30 +00:00
self . failUnlessRWChildURIIs ( self . public_root ,
u " foo " ,
self . _foo_uri ) )
2008-10-29 05:00:15 +00:00
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 )
2010-07-11 20:02:52 +00:00
d . addCallback ( lambda res : self . failUnlessReallyEqual ( res . strip ( ) , new_uri ) )
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 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 )
2010-07-11 20:02:52 +00:00
d . addCallback ( lambda res : self . failUnlessReallyEqual ( res . strip ( ) , new_uri ) )
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 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
2010-01-27 23:06:42 +00:00
def test_PUT_NEWFILEURL_uri_unknown_bad ( self ) :
2010-05-19 05:51:46 +00:00
d = self . PUT ( self . public_url + " /foo/put-future.txt?t=uri " , unknown_rwcap )
2010-01-27 23:06:42 +00:00
d . addBoth ( self . shouldFail , error . Error ,
" POST_put_uri_unknown_bad " ,
" 400 Bad Request " ,
" unknown cap in a write slot " )
return d
def test_PUT_NEWFILEURL_uri_unknown_ro_good ( self ) :
2010-05-19 05:51:46 +00:00
d = self . PUT ( self . public_url + " /foo/put-future-ro.txt?t=uri " , unknown_rocap )
2010-01-27 23:06:42 +00:00
d . addCallback ( self . failUnlessURIMatchesROChild , self . _foo_node ,
u " put-future-ro.txt " )
return d
def test_PUT_NEWFILEURL_uri_unknown_imm_good ( self ) :
2010-05-19 05:51:46 +00:00
d = self . PUT ( self . public_url + " /foo/put-future-imm.txt?t=uri " , unknown_immcap )
2010-01-27 23:06:42 +00:00
d . addCallback ( self . failUnlessURIMatchesROChild , self . _foo_node ,
u " put-future-imm.txt " )
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 )
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( FakeCHKFileNode . all_contents [ uri ] ,
file_contents )
2009-04-08 02:13:40 +00:00
return self . GET ( " /uri/ %s " % uri )
d . addCallback ( _check )
def _check2 ( res ) :
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( res , file_contents )
2009-04-08 02:13:40 +00:00
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 )
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( 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 ) :
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( res , file_contents )
2007-09-06 00:12:27 +00:00
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 )
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
def _check1 ( filecap ) :
filecap = filecap . strip ( )
self . failUnless ( filecap . startswith ( " URI:SSK: " ) , filecap )
self . filecap = filecap
u = uri . WriteableSSKFileURI . init_from_string ( filecap )
2010-02-22 02:45:04 +00:00
self . failUnless ( u . get_storage_index ( ) in FakeMutableFileNode . all_contents )
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
n = self . s . create_node_from_uri ( filecap )
2008-04-18 00:51:38 +00:00
return n . download_best_version ( )
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
d . addCallback ( _check1 )
def _check2 ( data ) :
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( data , file_contents )
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
return self . GET ( " /uri/ %s " % urllib . quote ( self . filecap ) )
2008-02-06 05:18:02 +00:00
d . addCallback ( _check2 )
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
def _check3 ( res ) :
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( res , file_contents )
Overhaul IFilesystemNode handling, to simplify tests and use POLA internally.
* stop using IURI as an adapter
* pass cap strings around instead of URI instances
* move filenode/dirnode creation duties from Client to new NodeMaker class
* move other Client duties to KeyGenerator, SecretHolder, History classes
* stop passing Client reference to dirnode/filenode constructors
- pass less-powerful references instead, like StorageBroker or Uploader
* always create DirectoryNodes by wrapping a filenode (mutable for now)
* remove some specialized mock classes from unit tests
Detailed list of changes (done one at a time, then merged together)
always pass a string to create_node_from_uri(), not an IURI instance
always pass a string to IFilesystemNode constructors, not an IURI instance
stop using IURI() as an adapter, switch on cap prefix in create_node_from_uri()
client.py: move SecretHolder code out to a separate class
test_web.py: hush pyflakes
client.py: move NodeMaker functionality out into a separate object
LiteralFileNode: stop storing a Client reference
immutable Checker: remove Client reference, it only needs a SecretHolder
immutable Upload: remove Client reference, leave SecretHolder and StorageBroker
immutable Repairer: replace Client reference with StorageBroker and SecretHolder
immutable FileNode: remove Client reference
mutable.Publish: stop passing Client
mutable.ServermapUpdater: get StorageBroker in constructor, not by peeking into Client reference
MutableChecker: reference StorageBroker and History directly, not through Client
mutable.FileNode: removed unused indirection to checker classes
mutable.FileNode: remove Client reference
client.py: move RSA key generation into a separate class, so it can be passed to the nodemaker
move create_mutable_file() into NodeMaker
test_dirnode.py: stop using FakeClient mockups, use NoNetworkGrid instead. This simplifies the code, but takes longer to run (17s instead of 6s). This should come down later when other cleanups make it possible to use simpler (non-RSA) fake mutable files for dirnode tests.
test_mutable.py: clean up basedir names
client.py: move create_empty_dirnode() into NodeMaker
dirnode.py: get rid of DirectoryNode.create
remove DirectoryNode.init_from_uri, refactor NodeMaker for customization, simplify test_web's mock Client to match
stop passing Client to DirectoryNode, make DirectoryNode.create_with_mutablefile the normal DirectoryNode constructor, start removing client from NodeMaker
remove Client from NodeMaker
move helper status into History, pass History to web.Status instead of Client
test_mutable.py: fix minor typo
2009-08-15 11:02:56 +00:00
d . addCallback ( _check3 )
2008-02-06 05:18:02 +00:00
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
Change OphandleTable to use a deterministic clock, so we can test it
To test the changes for #577, we need a deterministic way to simulate
the passage of long periods of time. twisted.internet.task.Clock seems,
from my Googling, to be the way to go for this functionality. I changed
a few things so that OphandleTable would use twisted.internet.task.Clock
when testing:
* WebishServer.__init___ now takes an optional 'clock' parameter,
* which it passes to the root.Root instance it creates.
* root.Root.__init__ now takes an optional 'clock' parameter, which it
passes to the OphandleTable.__init__ method.
* OphandleTable.__init__ now takes an optional 'clock' parameter. If
it is provided, and it isn't None, its callLater method will be used
to schedule ophandle expirations (as opposed to using
reactor.callLater, which is what OphandleTable does normally).
* The WebMixin object in test_web.py now sets a self.clock parameter,
which is a twisted.internet.task.Clock that it feeds to the
WebishServer it creates.
Tests using the WebMixin can control the passage of time in
OphandleTable by accessing self.clock.
2010-02-20 21:07:13 +00:00
d . addCallback ( lambda ign :
self . clock . advance ( 2.0 ) )
2008-10-22 05:13:54 +00:00
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
2010-02-21 01:04:55 +00:00
def test_uncollected_ophandle_expiration ( self ) :
# uncollected ophandles should expire after 4 days
def _make_uncollected_ophandle ( ophandle ) :
d = self . POST ( self . public_url +
" /foo/?t=start-manifest&ophandle= %d " % ophandle ,
followRedirect = False )
# When we start the operation, the webapi server will want
# to redirect us to the page for the ophandle, so we get
# confirmation that the operation has started. If the
# manifest operation has finished by the time we get there,
# following that redirect (by setting followRedirect=True
# above) has the side effect of collecting the ophandle that
# we've just created, which means that we can't use the
# ophandle to test the uncollected timeout anymore. So,
# instead, catch the 302 here and don't follow it.
d . addBoth ( self . should302 , " uncollected_ophandle_creation " )
return d
# Create an ophandle, don't collect it, then advance the clock by
# 4 days - 1 second and make sure that the ophandle is still there.
d = _make_uncollected_ophandle ( 131 )
d . addCallback ( lambda ign :
self . clock . advance ( ( 96 * 60 * 60 ) - 1 ) ) # 96 hours = 4 days
d . addCallback ( lambda ign :
self . GET ( " /operations/131?t=status&output=JSON " ) )
def _check1 ( res ) :
data = simplejson . loads ( res )
self . failUnless ( " finished " in data , res )
d . addCallback ( _check1 )
# Create an ophandle, don't collect it, then try to collect it
# after 4 days. It should be gone.
d . addCallback ( lambda ign :
_make_uncollected_ophandle ( 132 ) )
d . addCallback ( lambda ign :
self . clock . advance ( 96 * 60 * 60 ) )
d . addCallback ( lambda ign :
self . shouldHTTPError ( " test_uncollected_ophandle_expired_after_100_hours " ,
404 , " 404 Not Found " ,
" unknown/expired handle ' 132 ' " ,
self . GET ,
" /operations/132?t=status&output=JSON " ) )
return d
def test_collected_ophandle_expiration ( self ) :
# collected ophandles should expire after 1 day
def _make_collected_ophandle ( ophandle ) :
d = self . POST ( self . public_url +
" /foo/?t=start-manifest&ophandle= %d " % ophandle ,
followRedirect = True )
# By following the initial redirect, we collect the ophandle
# we've just created.
return d
# Create a collected ophandle, then collect it after 23 hours
# and 59 seconds to make sure that it is still there.
d = _make_collected_ophandle ( 133 )
d . addCallback ( lambda ign :
self . clock . advance ( ( 24 * 60 * 60 ) - 1 ) )
d . addCallback ( lambda ign :
self . GET ( " /operations/133?t=status&output=JSON " ) )
def _check1 ( res ) :
data = simplejson . loads ( res )
self . failUnless ( " finished " in data , res )
d . addCallback ( _check1 )
# Create another uncollected ophandle, then try to collect it
# after 24 hours to make sure that it is gone.
d . addCallback ( lambda ign :
_make_collected_ophandle ( 134 ) )
d . addCallback ( lambda ign :
self . clock . advance ( 24 * 60 * 60 ) )
d . addCallback ( lambda ign :
self . shouldHTTPError ( " test_collected_ophandle_expired_after_1000_minutes " ,
404 , " 404 Not Found " ,
" unknown/expired handle ' 134 ' " ,
self . GET ,
" /operations/134?t=status&output=JSON " ) )
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 ) :
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( res , " hello " )
2008-10-29 22:34:31 +00:00
d . addCallback ( _check )
return d
2008-10-22 00:52:56 +00:00
2010-07-11 20:02:52 +00:00
class Util ( ShouldFailMixin , testutil . ReallyEqualMixin , unittest . TestCase ) :
2010-05-10 06:37:29 +00:00
def test_load_file ( self ) :
# This will raise an exception unless a well-formed XML file is found under that name.
common . getxmlfile ( ' directory.xhtml ' ) . load ( )
2009-07-20 03:47:05 +00:00
def test_parse_replace_arg ( self ) :
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( common . parse_replace_arg ( " true " ) , True )
self . failUnlessReallyEqual ( common . parse_replace_arg ( " false " ) , False )
self . failUnlessReallyEqual ( common . parse_replace_arg ( " only-files " ) ,
" only-files " )
2009-07-20 03:47:05 +00:00
self . shouldFail ( AssertionError , " test_parse_replace_arg " , " " ,
common . parse_replace_arg , " only_fles " )
2008-05-20 22:21:46 +00:00
def test_abbreviate_time ( self ) :
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( common . abbreviate_time ( None ) , " " )
self . failUnlessReallyEqual ( common . abbreviate_time ( 1.234 ) , " 1.23s " )
self . failUnlessReallyEqual ( common . abbreviate_time ( 0.123 ) , " 123ms " )
self . failUnlessReallyEqual ( common . abbreviate_time ( 0.00123 ) , " 1.2ms " )
self . failUnlessReallyEqual ( common . abbreviate_time ( 0.000123 ) , " 123us " )
2010-07-29 21:06:38 +00:00
self . failUnlessReallyEqual ( common . abbreviate_time ( - 123000 ) , " -123000000000us " )
2008-05-20 22:21:46 +00:00
2010-08-15 14:19:33 +00:00
def test_compute_rate ( self ) :
self . failUnlessReallyEqual ( common . compute_rate ( None , None ) , None )
self . failUnlessReallyEqual ( common . compute_rate ( None , 1 ) , None )
self . failUnlessReallyEqual ( common . compute_rate ( 250000 , None ) , None )
self . failUnlessReallyEqual ( common . compute_rate ( 250000 , 0 ) , None )
self . failUnlessReallyEqual ( common . compute_rate ( 250000 , 10 ) , 25000.0 )
self . failUnlessReallyEqual ( common . compute_rate ( 0 , 10 ) , 0.0 )
self . shouldFail ( AssertionError , " test_compute_rate " , " " ,
common . compute_rate , - 100 , 10 )
self . shouldFail ( AssertionError , " test_compute_rate " , " " ,
common . compute_rate , 100 , - 10 )
# Sanity check
rate = common . compute_rate ( 10 * 1000 * 1000 , 1 )
self . failUnlessReallyEqual ( common . abbreviate_rate ( rate ) , " 10.00MBps " )
2008-05-20 22:21:46 +00:00
def test_abbreviate_rate ( self ) :
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( common . abbreviate_rate ( None ) , " " )
self . failUnlessReallyEqual ( common . abbreviate_rate ( 1234000 ) , " 1.23MBps " )
self . failUnlessReallyEqual ( common . abbreviate_rate ( 12340 ) , " 12.3kBps " )
self . failUnlessReallyEqual ( common . abbreviate_rate ( 123 ) , " 123Bps " )
2008-05-20 22:21:46 +00:00
def test_abbreviate_size ( self ) :
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( common . abbreviate_size ( None ) , " " )
self . failUnlessReallyEqual ( common . abbreviate_size ( 1.23 * 1000 * 1000 * 1000 ) , " 1.23GB " )
self . failUnlessReallyEqual ( common . abbreviate_size ( 1.23 * 1000 * 1000 ) , " 1.23MB " )
self . failUnlessReallyEqual ( common . abbreviate_size ( 1230 ) , " 1.2kB " )
self . failUnlessReallyEqual ( common . abbreviate_size ( 123 ) , " 123B " )
2008-05-20 22:21:46 +00:00
2009-02-22 01:01:16 +00:00
def test_plural ( self ) :
def convert ( s ) :
return " %d second %s " % ( s , status . plural ( s ) )
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( convert ( 0 ) , " 0 seconds " )
self . failUnlessReallyEqual ( convert ( 1 ) , " 1 second " )
self . failUnlessReallyEqual ( convert ( 2 ) , " 2 seconds " )
2009-02-22 01:01:16 +00:00
def convert2 ( s ) :
return " has share %s : %s " % ( status . plural ( s ) , " , " . join ( s ) )
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( convert2 ( [ ] ) , " has shares: " )
self . failUnlessReallyEqual ( convert2 ( [ " 1 " ] ) , " has share: 1 " )
self . failUnlessReallyEqual ( convert2 ( [ " 1 " , " 2 " ] ) , " has shares: 1,2 " )
2009-02-22 01:01:16 +00:00
2009-02-17 05:12:42 +00:00
2010-07-11 20:02:52 +00:00
class Grid ( GridTestMixin , WebErrorMixin , ShouldFailMixin , testutil . ReallyEqualMixin , unittest . TestCase ) :
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 " )
2009-11-26 23:27:31 +00:00
d . addCallback ( lambda ign : c0 . create_immutable_dirnode ( { } ) )
d . addCallback ( _stash_mutable_uri , " smalldir " )
2009-02-17 05:12:42 +00:00
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 ) :
2010-07-19 04:50:47 +00:00
good_shares = self . find_uri_shares ( self . uris [ " good " ] )
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( len ( good_shares ) , 10 )
2010-07-19 04:50:47 +00:00
sick_shares = self . find_uri_shares ( self . uris [ " sick " ] )
2009-02-17 05:12:42 +00:00
os . unlink ( sick_shares [ 0 ] [ 2 ] )
2010-07-19 04:50:47 +00:00
dead_shares = self . find_uri_shares ( self . uris [ " dead " ] )
2009-02-17 05:12:42 +00:00
for i in range ( 1 , 10 ) :
os . unlink ( dead_shares [ i ] [ 2 ] )
2010-07-19 04:50:47 +00:00
c_shares = self . find_uri_shares ( self . uris [ " corrupt " ] )
2009-02-17 05:12:42 +00:00
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 )
2010-07-18 14:29:15 +00:00
self . failUnlessEqual ( r [ " summary " ] , " Healthy " )
2009-02-17 05:12:42 +00:00
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 )
2010-07-18 14:29:15 +00:00
self . failUnlessEqual ( r [ " storage-index " ] , " " )
2009-02-17 05:12:42 +00:00
self . failUnless ( r [ " results " ] [ " healthy " ] )
d . addCallback ( _got_json_small )
2009-11-26 23:27:31 +00:00
d . addCallback ( self . CHECK , " smalldir " , " t=check " )
def _got_html_smalldir ( res ) :
self . failUnless ( " Literal files are always healthy " in res , res )
self . failIf ( " Not Healthy " in res , res )
d . addCallback ( _got_html_smalldir )
d . addCallback ( self . CHECK , " smalldir " , " t=check&output=json " )
def _got_json_smalldir ( res ) :
r = simplejson . loads ( res )
2010-07-18 14:29:15 +00:00
self . failUnlessEqual ( r [ " storage-index " ] , " " )
2009-11-26 23:27:31 +00:00
self . failUnless ( r [ " results " ] [ " healthy " ] )
d . addCallback ( _got_json_smalldir )
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 )
2010-07-18 14:29:15 +00:00
self . failUnlessEqual ( r [ " summary " ] ,
" Not Healthy: 9 shares (enc 3-of-10) " )
2009-02-17 05:12:42 +00:00
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 )
2010-07-18 14:29:15 +00:00
self . failUnlessEqual ( r [ " summary " ] ,
" Not Healthy: 1 shares (enc 3-of-10) " )
2009-02-17 05:12:42 +00:00
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 " ] )
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( r [ " results " ] [ " count-shares-good " ] , 9 )
self . failUnlessReallyEqual ( r [ " results " ] [ " count-corrupt-shares " ] , 1 )
2009-02-17 05:12:42 +00:00
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 ) :
2010-07-19 04:50:47 +00:00
good_shares = self . find_uri_shares ( self . uris [ " good " ] )
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( len ( good_shares ) , 10 )
2010-07-19 04:50:47 +00:00
sick_shares = self . find_uri_shares ( self . uris [ " sick " ] )
2009-02-17 05:12:42 +00:00
os . unlink ( sick_shares [ 0 ] [ 2 ] )
2010-07-19 04:50:47 +00:00
dead_shares = self . find_uri_shares ( self . uris [ " dead " ] )
2009-02-17 05:12:42 +00:00
for i in range ( 1 , 10 ) :
os . unlink ( dead_shares [ i ] [ 2 ] )
2010-07-19 04:50:47 +00:00
c_shares = self . find_uri_shares ( self . uris [ " corrupt " ] )
2009-02-17 05:12:42 +00:00
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 ) :
2010-07-19 04:50:47 +00:00
sick_shares = self . find_uri_shares ( self . uris [ " sick " ] )
2009-02-17 05:12:42 +00:00
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 )
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( r [ " repair-attempted " ] , True )
self . failUnlessReallyEqual ( r [ " repair-successful " ] , True )
2010-07-18 14:29:15 +00:00
self . failUnlessEqual ( r [ " pre-repair-results " ] [ " summary " ] ,
" Not Healthy: 9 shares (enc 3-of-10) " )
2009-02-17 05:12:42 +00:00
self . failIf ( r [ " pre-repair-results " ] [ " results " ] [ " healthy " ] )
2010-07-18 14:29:15 +00:00
self . failUnlessEqual ( r [ " post-repair-results " ] [ " summary " ] , " healthy " )
2009-02-17 05:12:42 +00:00
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
2010-01-27 06:44:30 +00:00
def test_unknown ( self , immutable = False ) :
2009-07-03 01:07:49 +00:00
self . basedir = " web/Grid/unknown "
2010-01-27 06:44:30 +00:00
if immutable :
self . basedir = " web/Grid/unknown-immutable "
2009-07-03 01:07:49 +00:00
self . set_up_grid ( )
c0 = self . g . clients [ 0 ]
self . uris = { }
self . fileurls = { }
# the future cap format may contain slashes, which must be tolerated
2010-05-19 05:51:46 +00:00
expected_info_url = " uri/ %s ?t=info " % urllib . quote ( unknown_rwcap ,
2009-07-03 01:07:49 +00:00
safe = " " )
2010-01-27 06:44:30 +00:00
if immutable :
name = u " future-imm "
2010-05-19 05:51:46 +00:00
future_node = UnknownNode ( None , unknown_immcap , deep_immutable = True )
2010-01-27 06:44:30 +00:00
d = c0 . create_immutable_dirnode ( { name : ( future_node , { } ) } )
else :
name = u " future "
2010-05-19 05:51:46 +00:00
future_node = UnknownNode ( unknown_rwcap , unknown_rocap )
2010-01-27 06:44:30 +00:00
d = c0 . create_dirnode ( )
2009-07-03 01:07:49 +00:00
def _stash_root_and_create_file ( n ) :
self . rootnode = n
self . rooturl = " uri/ " + urllib . quote ( n . get_uri ( ) ) + " / "
self . rourl = " uri/ " + urllib . quote ( n . get_readonly_uri ( ) ) + " / "
2010-01-27 06:44:30 +00:00
if not immutable :
return self . rootnode . set_node ( name , future_node )
2009-07-03 01:07:49 +00:00
d . addCallback ( _stash_root_and_create_file )
2010-01-27 06:44:30 +00:00
2009-07-03 01:07:49 +00:00
# make sure directory listing tolerates unknown nodes
d . addCallback ( lambda ign : self . GET ( self . rooturl ) )
2010-01-28 22:08:00 +00:00
def _check_directory_html ( res , expected_type_suffix ) :
pattern = re . compile ( r ' <td> \ ? %s </td>[ \ t \ n \ r]* '
' <td> %s </td> ' % ( expected_type_suffix , str ( name ) ) ,
re . DOTALL )
self . failUnless ( re . search ( pattern , res ) , res )
2010-01-27 06:44:30 +00:00
# find the More Info link for name, should be relative
2009-07-03 01:07:49 +00:00
mo = re . search ( r ' <a href= " ([^ " ]+) " >More Info</a> ' , res )
info_url = mo . group ( 1 )
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( info_url , " %s ?t=info " % ( str ( name ) , ) )
2010-01-29 03:52:10 +00:00
if immutable :
d . addCallback ( _check_directory_html , " -IMM " )
else :
d . addCallback ( _check_directory_html , " " )
2009-07-03 01:07:49 +00:00
d . addCallback ( lambda ign : self . GET ( self . rooturl + " ?t=json " ) )
2010-01-27 06:44:30 +00:00
def _check_directory_json ( res , expect_rw_uri ) :
2009-07-03 01:07:49 +00:00
data = simplejson . loads ( res )
2010-07-18 14:29:15 +00:00
self . failUnlessEqual ( data [ 0 ] , " dirnode " )
2010-01-27 06:44:30 +00:00
f = data [ 1 ] [ " children " ] [ name ]
2010-07-18 14:29:15 +00:00
self . failUnlessEqual ( f [ 0 ] , " unknown " )
2010-01-27 06:44:30 +00:00
if expect_rw_uri :
2010-07-18 14:29:15 +00:00
self . failUnlessReallyEqual ( to_str ( f [ 1 ] [ " rw_uri " ] ) , unknown_rwcap , data )
2009-07-03 01:07:49 +00:00
else :
self . failIfIn ( " rw_uri " , f [ 1 ] )
2010-01-29 03:52:10 +00:00
if immutable :
2010-07-18 14:29:15 +00:00
self . failUnlessReallyEqual ( to_str ( f [ 1 ] [ " ro_uri " ] ) , unknown_immcap , data )
2010-01-29 03:52:10 +00:00
else :
2010-07-18 14:29:15 +00:00
self . failUnlessReallyEqual ( to_str ( f [ 1 ] [ " ro_uri " ] ) , unknown_rocap , data )
2009-07-03 01:07:49 +00:00
self . failUnless ( " metadata " in f [ 1 ] )
2010-01-27 06:44:30 +00:00
d . addCallback ( _check_directory_json , expect_rw_uri = not immutable )
def _check_info ( res , expect_rw_uri , expect_ro_uri ) :
2009-07-03 01:07:49 +00:00
self . failUnlessIn ( " Object Type: <span>unknown</span> " , res )
2010-01-27 06:44:30 +00:00
if expect_rw_uri :
2010-05-19 05:51:46 +00:00
self . failUnlessIn ( unknown_rwcap , res )
2010-01-27 06:44:30 +00:00
if expect_ro_uri :
2010-05-19 05:51:46 +00:00
if immutable :
self . failUnlessIn ( unknown_immcap , res )
else :
self . failUnlessIn ( unknown_rocap , res )
2010-01-27 06:44:30 +00:00
else :
2010-05-19 05:51:46 +00:00
self . failIfIn ( unknown_rocap , res )
2009-07-03 01:07:49 +00:00
self . failIfIn ( " Raw data as " , res )
self . failIfIn ( " Directory writecap " , res )
self . failIfIn ( " Checker Operations " , res )
self . failIfIn ( " Mutable File Operations " , res )
self . failIfIn ( " Directory Operations " , res )
2010-01-27 06:44:30 +00:00
# FIXME: these should have expect_rw_uri=not immutable; I don't know
# why they fail. Possibly related to ticket #922.
d . addCallback ( lambda ign : self . GET ( expected_info_url ) )
d . addCallback ( _check_info , expect_rw_uri = False , expect_ro_uri = False )
d . addCallback ( lambda ign : self . GET ( " %s %s ?t=info " % ( self . rooturl , str ( name ) ) ) )
d . addCallback ( _check_info , expect_rw_uri = False , expect_ro_uri = True )
def _check_json ( res , expect_rw_uri ) :
data = simplejson . loads ( res )
2010-07-18 14:29:15 +00:00
self . failUnlessEqual ( data [ 0 ] , " unknown " )
2010-01-27 06:44:30 +00:00
if expect_rw_uri :
2010-07-18 14:29:15 +00:00
self . failUnlessReallyEqual ( to_str ( data [ 1 ] [ " rw_uri " ] ) , unknown_rwcap , data )
2010-01-27 06:44:30 +00:00
else :
self . failIfIn ( " rw_uri " , data [ 1 ] )
2010-01-29 03:52:10 +00:00
if immutable :
2010-07-18 14:29:15 +00:00
self . failUnlessReallyEqual ( to_str ( data [ 1 ] [ " ro_uri " ] ) , unknown_immcap , data )
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( data [ 1 ] [ " mutable " ] , False )
2010-01-29 03:52:10 +00:00
elif expect_rw_uri :
2010-07-18 14:29:15 +00:00
self . failUnlessReallyEqual ( to_str ( data [ 1 ] [ " ro_uri " ] ) , unknown_rocap , data )
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( data [ 1 ] [ " mutable " ] , True )
2010-01-29 03:52:10 +00:00
else :
2010-07-18 14:29:15 +00:00
self . failUnlessReallyEqual ( to_str ( data [ 1 ] [ " ro_uri " ] ) , unknown_rocap , data )
2010-01-29 03:52:10 +00:00
self . failIf ( " mutable " in data [ 1 ] , data [ 1 ] )
2010-01-27 06:44:30 +00:00
# TODO: check metadata contents
self . failUnless ( " metadata " in data [ 1 ] )
d . addCallback ( lambda ign : self . GET ( " %s %s ?t=json " % ( self . rooturl , str ( name ) ) ) )
d . addCallback ( _check_json , expect_rw_uri = not immutable )
2009-07-03 01:07:49 +00:00
# and make sure that a read-only version of the directory can be
2010-05-19 05:51:46 +00:00
# rendered too. This version will not have unknown_rwcap, whether
2010-01-27 06:44:30 +00:00
# or not future_node was immutable.
2009-07-03 01:07:49 +00:00
d . addCallback ( lambda ign : self . GET ( self . rourl ) )
2010-01-29 03:52:10 +00:00
if immutable :
d . addCallback ( _check_directory_html , " -IMM " )
else :
d . addCallback ( _check_directory_html , " -RO " )
2009-07-03 01:07:49 +00:00
d . addCallback ( lambda ign : self . GET ( self . rourl + " ?t=json " ) )
2010-01-27 06:44:30 +00:00
d . addCallback ( _check_directory_json , expect_rw_uri = False )
d . addCallback ( lambda ign : self . GET ( " %s %s ?t=json " % ( self . rourl , str ( name ) ) ) )
d . addCallback ( _check_json , expect_rw_uri = False )
# TODO: check that getting t=info from the Info link in the ro directory
# works, and does not include the writecap URI.
return d
def test_immutable_unknown ( self ) :
return self . test_unknown ( immutable = True )
def test_mutant_dirnodes_are_omitted ( self ) :
self . basedir = " web/Grid/mutant_dirnodes_are_omitted "
self . set_up_grid ( )
c = self . g . clients [ 0 ]
nm = c . nodemaker
self . uris = { }
self . fileurls = { }
lonely_uri = " URI:LIT:n5xgk " # LIT for "one"
mut_write_uri = " URI:SSK:vfvcbdfbszyrsaxchgevhmmlii:euw4iw7bbnkrrwpzuburbhppuxhc3gwxv26f6imekhz7zyw2ojnq "
mut_read_uri = " URI:SSK-RO:e3mdrzfwhoq42hy5ubcz6rp3o4:ybyibhnp3vvwuq2vaw2ckjmesgkklfs6ghxleztqidihjyofgw7q "
# This method tests mainly dirnode, but we'd have to duplicate code in order to
# test the dirnode and web layers separately.
# 'lonely' is a valid LIT child, 'ro' is a mutant child with an SSK-RO readcap,
# and 'write-in-ro' is a mutant child with an SSK writecap in the ro_uri field.
# When the directory is read, the mutants should be silently disposed of, leaving
# their lonely sibling.
# We don't test the case of a retrieving a cap from the encrypted rw_uri field,
# because immutable directories don't have a writecap and therefore that field
# isn't (and can't be) decrypted.
# TODO: The field still exists in the netstring. Technically we should check what
2010-01-28 20:24:56 +00:00
# happens if something is put there (_unpack_contents should raise ValueError),
# but that can wait.
2010-01-27 06:44:30 +00:00
lonely_child = nm . create_from_cap ( lonely_uri )
mutant_ro_child = nm . create_from_cap ( mut_read_uri )
mutant_write_in_ro_child = nm . create_from_cap ( mut_write_uri )
def _by_hook_or_by_crook ( ) :
return True
for n in [ mutant_ro_child , mutant_write_in_ro_child ] :
n . is_allowed_in_immutable_directory = _by_hook_or_by_crook
mutant_write_in_ro_child . get_write_uri = lambda : None
mutant_write_in_ro_child . get_readonly_uri = lambda : mut_write_uri
kids = { u " lonely " : ( lonely_child , { } ) ,
u " ro " : ( mutant_ro_child , { } ) ,
u " write-in-ro " : ( mutant_write_in_ro_child , { } ) ,
}
d = c . create_immutable_dirnode ( kids )
def _created ( dn ) :
self . failUnless ( isinstance ( dn , dirnode . DirectoryNode ) )
self . failIf ( dn . is_mutable ( ) )
self . failUnless ( dn . is_readonly ( ) )
# This checks that if we somehow ended up calling dn._decrypt_rwcapdata, it would fail.
self . failIf ( hasattr ( dn . _node , ' get_writekey ' ) )
rep = str ( dn )
self . failUnless ( " RO-IMM " in rep )
cap = dn . get_cap ( )
self . failUnlessIn ( " CHK " , cap . to_string ( ) )
self . cap = cap
self . rootnode = dn
self . rooturl = " uri/ " + urllib . quote ( dn . get_uri ( ) ) + " / "
return download_to_data ( dn . _node )
d . addCallback ( _created )
def _check_data ( data ) :
# Decode the netstring representation of the directory to check that all children
# are present. This is a bit of an abstraction violation, but there's not really
# any other way to do it given that the real DirectoryNode._unpack_contents would
# strip the mutant children out (which is what we're trying to test, later).
position = 0
numkids = 0
while position < len ( data ) :
entries , position = split_netstring ( data , 1 , position )
entry = entries [ 0 ]
2010-01-27 23:06:42 +00:00
( name_utf8 , ro_uri , rwcapdata , metadata_s ) , subpos = split_netstring ( entry , 4 )
name = name_utf8 . decode ( " utf-8 " )
2010-01-27 06:44:30 +00:00
self . failUnless ( rwcapdata == " " )
2010-01-27 23:06:42 +00:00
self . failUnless ( name in kids )
( expected_child , ign ) = kids [ name ]
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( ro_uri , expected_child . get_readonly_uri ( ) )
2010-01-27 23:06:42 +00:00
numkids + = 1
2010-01-27 06:44:30 +00:00
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( numkids , 3 )
2010-01-27 06:44:30 +00:00
return self . rootnode . list ( )
d . addCallback ( _check_data )
# Now when we use the real directory listing code, the mutants should be absent.
def _check_kids ( children ) :
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( sorted ( children . keys ( ) ) , [ u " lonely " ] )
2010-01-27 06:44:30 +00:00
lonely_node , lonely_metadata = children [ u " lonely " ]
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( lonely_node . get_write_uri ( ) , None )
self . failUnlessReallyEqual ( lonely_node . get_readonly_uri ( ) , lonely_uri )
2010-01-27 06:44:30 +00:00
d . addCallback ( _check_kids )
d . addCallback ( lambda ign : nm . create_from_cap ( self . cap . to_string ( ) ) )
d . addCallback ( lambda n : n . list ( ) )
d . addCallback ( _check_kids ) # again with dirnode recreated from cap
# Make sure the lonely child can be listed in HTML...
d . addCallback ( lambda ign : self . GET ( self . rooturl ) )
def _check_html ( res ) :
self . failIfIn ( " URI:SSK " , res )
get_lonely = " " . join ( [ r ' <td>FILE</td> ' ,
r ' \ s+<td> ' ,
r ' <a href= " [^ " ]+ %s [^ " ]+ " >lonely</a> ' % ( urllib . quote ( lonely_uri ) , ) ,
r ' </td> ' ,
2011-06-11 16:37:41 +00:00
r ' \ s+<td align= " right " > %d </td> ' % len ( " one " ) ,
2010-01-27 06:44:30 +00:00
] )
self . failUnless ( re . search ( get_lonely , res ) , res )
# find the More Info link for name, should be relative
mo = re . search ( r ' <a href= " ([^ " ]+) " >More Info</a> ' , res )
info_url = mo . group ( 1 )
self . failUnless ( info_url . endswith ( urllib . quote ( lonely_uri ) + " ?t=info " ) , info_url )
d . addCallback ( _check_html )
# ... and in JSON.
d . addCallback ( lambda ign : self . GET ( self . rooturl + " ?t=json " ) )
def _check_json ( res ) :
data = simplejson . loads ( res )
2010-07-18 14:29:15 +00:00
self . failUnlessEqual ( data [ 0 ] , " dirnode " )
2010-01-27 06:44:30 +00:00
listed_children = data [ 1 ] [ " children " ]
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( sorted ( listed_children . keys ( ) ) , [ u " lonely " ] )
2010-01-27 06:44:30 +00:00
ll_type , ll_data = listed_children [ u " lonely " ]
2010-07-18 14:29:15 +00:00
self . failUnlessEqual ( ll_type , " filenode " )
2010-01-27 06:44:30 +00:00
self . failIf ( " rw_uri " in ll_data )
2010-07-18 14:29:15 +00:00
self . failUnlessReallyEqual ( to_str ( ll_data [ " ro_uri " ] ) , lonely_uri )
2010-01-27 06:44:30 +00:00
d . addCallback ( _check_json )
2009-07-03 01:07:49 +00:00
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
2009-10-12 22:45:06 +00:00
d = c0 . create_dirnode ( )
2009-02-17 06:35:53 +00:00
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 " )
2009-07-03 01:07:49 +00:00
# this tests that deep-check and stream-manifest will ignore
# UnknownNode instances. Hopefully this will also cover deep-stats.
2010-05-19 05:51:46 +00:00
future_node = UnknownNode ( unknown_rwcap , unknown_rocap )
2010-01-27 06:44:30 +00:00
d . addCallback ( lambda ign : self . rootnode . set_node ( u " future " , future_node ) )
2009-07-03 01:07:49 +00:00
2009-02-24 22:40:17 +00:00
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-07-03 01:07:49 +00:00
# root/future
2009-02-17 06:35:53 +00:00
d . addCallback ( self . CHECK , " root " , " t=stream-deep-check " )
def _done ( res ) :
2009-07-03 01:07:49 +00:00
try :
units = [ simplejson . loads ( line )
for line in res . splitlines ( )
if line ]
except ValueError :
print " response is: " , res
print " undecodeable line was ' %s ' " % line
raise
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( len ( units ) , 5 + 1 )
2009-02-17 06:35:53 +00:00
# should be parent-first
u0 = units [ 0 ]
2010-07-18 14:29:15 +00:00
self . failUnlessEqual ( u0 [ " path " ] , [ ] )
self . failUnlessEqual ( u0 [ " type " ] , " directory " )
self . failUnlessReallyEqual ( to_str ( u0 [ " cap " ] ) , self . rootnode . get_uri ( ) )
2009-02-17 06:35:53 +00:00
u0cr = u0 [ " check-results " ]
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( u0cr [ " results " ] [ " count-shares-good " ] , 10 )
2009-02-17 06:35:53 +00:00
ugood = [ u for u in units
if u [ " type " ] == " file " and u [ " path " ] == [ u " good " ] ] [ 0 ]
2010-07-18 14:29:15 +00:00
self . failUnlessReallyEqual ( to_str ( ugood [ " cap " ] ) , self . uris [ " good " ] )
2009-02-17 06:35:53 +00:00
ugoodcr = ugood [ " check-results " ]
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( ugoodcr [ " results " ] [ " count-shares-good " ] , 10 )
2009-02-17 06:35:53 +00:00
stats = units [ - 1 ]
2010-07-18 14:29:15 +00:00
self . failUnlessEqual ( stats [ " type " ] , " stats " )
2009-02-17 06:35:53 +00:00
s = stats [ " stats " ]
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( s [ " count-immutable-files " ] , 2 )
self . failUnlessReallyEqual ( s [ " count-literal-files " ] , 1 )
self . failUnlessReallyEqual ( s [ " count-directories " ] , 1 )
self . failUnlessReallyEqual ( s [ " count-unknown " ] , 1 )
2009-02-17 06:35:53 +00:00
d . addCallback ( _done )
2009-07-03 01:07:49 +00:00
d . addCallback ( self . CHECK , " root " , " t=stream-manifest " )
def _check_manifest ( res ) :
self . failUnless ( res . endswith ( " \n " ) )
units = [ simplejson . loads ( t ) for t in res [ : - 1 ] . split ( " \n " ) ]
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( len ( units ) , 5 + 1 )
2010-07-18 14:29:15 +00:00
self . failUnlessEqual ( units [ - 1 ] [ " type " ] , " stats " )
2009-07-03 01:07:49 +00:00
first = units [ 0 ]
2010-07-18 14:29:15 +00:00
self . failUnlessEqual ( first [ " path " ] , [ ] )
self . failUnlessEqual ( to_str ( first [ " cap " ] ) , self . rootnode . get_uri ( ) )
self . failUnlessEqual ( first [ " type " ] , " directory " )
2009-07-03 01:07:49 +00:00
stats = units [ - 1 ] [ " stats " ]
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( stats [ " count-immutable-files " ] , 2 )
self . failUnlessReallyEqual ( stats [ " count-literal-files " ] , 1 )
self . failUnlessReallyEqual ( stats [ " count-mutable-files " ] , 0 )
self . failUnlessReallyEqual ( stats [ " count-immutable-files " ] , 2 )
self . failUnlessReallyEqual ( stats [ " count-unknown " ] , 1 )
2009-07-03 01:07:49 +00:00
d . addCallback ( _check_manifest )
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 :
2009-10-13 02:15:20 +00:00
self . rootnode . create_subdirectory ( u " subdir " ) )
2009-02-24 22:40:17 +00:00
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
2009-07-03 01:07:49 +00:00
# root/future
2009-02-25 06:13:35 +00:00
# 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 ] ]
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( len ( units ) , 6 ) # includes subdir
2009-02-25 06:13:35 +00:00
last_unit = units [ - 1 ]
2010-07-18 14:29:15 +00:00
self . failUnlessEqual ( last_unit [ " path " ] , [ " subdir " ] )
2009-02-25 06:13:35 +00:00
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 ] ]
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( len ( units ) , 6 ) # includes subdir
2009-02-25 06:13:35 +00:00
last_unit = units [ - 1 ]
2010-07-18 14:29:15 +00:00
self . failUnlessEqual ( last_unit [ " path " ] , [ " subdir " ] )
2009-02-25 06:13:35 +00:00
r = last_unit [ " check-results " ] [ " results " ]
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( r [ " count-recoverable-versions " ] , 0 )
self . failUnlessReallyEqual ( r [ " count-shares-good " ] , 1 )
self . failUnlessReallyEqual ( r [ " recoverable " ] , False )
2009-02-25 06:13:35 +00:00
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
2009-10-12 22:45:06 +00:00
d = c0 . create_dirnode ( )
2009-02-17 06:35:53 +00:00
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 ) :
2010-07-19 04:50:47 +00:00
good_shares = self . find_uri_shares ( self . uris [ " good " ] )
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( len ( good_shares ) , 10 )
2010-07-19 04:50:47 +00:00
sick_shares = self . find_uri_shares ( self . uris [ " sick " ] )
2009-02-17 06:35:53 +00:00
os . unlink ( sick_shares [ 0 ] [ 2 ] )
2010-07-19 04:50:47 +00:00
#dead_shares = self.find_uri_shares(self.uris["dead"])
2009-02-17 06:35:53 +00:00
#for i in range(1, 10):
# os.unlink(dead_shares[i][2])
2010-07-19 04:50:47 +00:00
#c_shares = self.find_uri_shares(self.uris["corrupt"])
2009-02-17 06:35:53 +00:00
#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 ]
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( len ( units ) , 4 + 1 )
2009-02-17 06:35:53 +00:00
# should be parent-first
u0 = units [ 0 ]
2010-07-18 14:29:15 +00:00
self . failUnlessEqual ( u0 [ " path " ] , [ ] )
self . failUnlessEqual ( u0 [ " type " ] , " directory " )
self . failUnlessReallyEqual ( to_str ( u0 [ " cap " ] ) , self . rootnode . get_uri ( ) )
2009-02-17 06:35:53 +00:00
u0crr = u0 [ " check-and-repair-results " ]
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( u0crr [ " repair-attempted " ] , False )
self . failUnlessReallyEqual ( u0crr [ " pre-repair-results " ] [ " results " ] [ " count-shares-good " ] , 10 )
2009-02-17 06:35:53 +00:00
ugood = [ u for u in units
if u [ " type " ] == " file " and u [ " path " ] == [ u " good " ] ] [ 0 ]
2010-07-18 14:29:15 +00:00
self . failUnlessEqual ( to_str ( ugood [ " cap " ] ) , self . uris [ " good " ] )
2009-02-17 06:35:53 +00:00
ugoodcrr = ugood [ " check-and-repair-results " ]
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( ugoodcrr [ " repair-attempted " ] , False )
self . failUnlessReallyEqual ( ugoodcrr [ " pre-repair-results " ] [ " results " ] [ " count-shares-good " ] , 10 )
2009-02-17 06:35:53 +00:00
usick = [ u for u in units
if u [ " type " ] == " file " and u [ " path " ] == [ u " sick " ] ] [ 0 ]
2010-07-18 14:29:15 +00:00
self . failUnlessReallyEqual ( to_str ( usick [ " cap " ] ) , self . uris [ " sick " ] )
2009-02-17 06:35:53 +00:00
usickcrr = usick [ " check-and-repair-results " ]
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( usickcrr [ " repair-attempted " ] , True )
self . failUnlessReallyEqual ( usickcrr [ " repair-successful " ] , True )
self . failUnlessReallyEqual ( usickcrr [ " pre-repair-results " ] [ " results " ] [ " count-shares-good " ] , 9 )
self . failUnlessReallyEqual ( usickcrr [ " post-repair-results " ] [ " results " ] [ " count-shares-good " ] , 10 )
2009-02-17 06:35:53 +00:00
stats = units [ - 1 ]
2010-07-18 14:29:15 +00:00
self . failUnlessEqual ( stats [ " type " ] , " stats " )
2009-02-17 06:35:53 +00:00
s = stats [ " stats " ]
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( s [ " count-immutable-files " ] , 2 )
self . failUnlessReallyEqual ( s [ " count-literal-files " ] , 1 )
self . failUnlessReallyEqual ( s [ " count-directories " ] , 1 )
2009-02-17 06:35:53 +00:00
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 ]
2010-07-19 04:50:47 +00:00
shares = self . find_uri_shares ( u )
2009-02-18 02:32:43 +00:00
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
2009-10-12 22:45:06 +00:00
d = c0 . create_dirnode ( )
2009-02-18 02:32:43 +00:00
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
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( len ( units ) , 4 + 1 )
2009-02-18 02:32:43 +00:00
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 ]
2010-01-07 19:13:25 +00:00
c0 . DEFAULT_ENCODING_PARAMETERS [ ' happy ' ] = 2
2009-03-04 04:56:30 +00:00
self . fileurls = { }
DATA = " data " * 100
2009-10-12 22:45:06 +00:00
d = c0 . create_dirnode ( )
2009-03-04 04:56:30 +00:00
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-10-12 22:45:06 +00:00
d . addCallback ( lambda ign : c0 . create_dirnode ( ) )
2009-03-07 11:57:07 +00:00
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 )
2009-10-12 22:45:06 +00:00
d . addCallback ( lambda ign : c0 . create_dirnode ( ) )
2009-03-07 11:57:07 +00:00
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: "
2010-08-04 07:27:10 +00:00
" no shares (need 3). Last failure: None " )
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( exp , body )
2009-03-04 04:56:30 +00:00
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 ( ) )
2010-09-01 01:37:02 +00:00
msgbase = ( " NotEnoughSharesError: This indicates that some "
" servers were unavailable, or that shares have been "
" lost to server departure, hard drive failure, or disk "
" corruption. You should perform a filecheck on "
" this object to learn more. The full error message is: "
)
msg1 = msgbase + ( " ran out of shares: "
" complete=sh0 "
" pending= "
" overdue= unused= need 3. Last failure: None " )
msg2 = msgbase + ( " ran out of shares: "
" complete= "
" pending=Share(sh0-on-xgru5) "
" overdue= unused= need 3. Last failure: None " )
2010-08-04 07:27:10 +00:00
self . failUnless ( body == msg1 or body == msg2 , body )
2009-03-04 04:56:30 +00:00
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. " )
2010-07-11 20:02:52 +00:00
self . failUnlessReallyEqual ( exp , body )
2009-03-07 11:57:07 +00:00
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 ( ) )
2009-12-27 19:58:28 +00:00
# "Accept: */*" : should get a text/html stack trace
# "Accept: text/plain" : should get a text/plain stack trace
# "Accept: text/plain, application/octet-stream" : text/plain (CLI)
# no Accept header: should get a text/html stack trace
2009-03-04 04:56:30 +00:00
d . addCallback ( lambda ignored :
self . shouldHTTPError ( " GET errorboom_html " ,
500 , " Internal Server Error " , None ,
2009-12-27 19:58:28 +00:00
self . GET , " ERRORBOOM " ,
headers = { " accept " : [ " */* " ] } ) )
def _internal_error_html1 ( body ) :
2009-03-04 04:56:30 +00:00
self . failUnless ( " <html> " in body , " expected HTML, not ' %s ' " % body )
2009-12-27 19:58:28 +00:00
d . addCallback ( _internal_error_html1 )
2009-03-04 04:56:30 +00:00
d . addCallback ( lambda ignored :
self . shouldHTTPError ( " GET errorboom_text " ,
500 , " Internal Server Error " , None ,
self . GET , " ERRORBOOM " ,
headers = { " accept " : [ " text/plain " ] } ) )
2009-12-27 19:58:28 +00:00
def _internal_error_text2 ( body ) :
self . failIf ( " <html> " in body , body )
self . failUnless ( body . startswith ( " Traceback " ) , body )
d . addCallback ( _internal_error_text2 )
CLI_accepts = " text/plain, application/octet-stream "
d . addCallback ( lambda ignored :
self . shouldHTTPError ( " GET errorboom_text " ,
500 , " Internal Server Error " , None ,
self . GET , " ERRORBOOM " ,
headers = { " accept " : [ CLI_accepts ] } ) )
def _internal_error_text3 ( body ) :
2009-03-04 04:56:30 +00:00
self . failIf ( " <html> " in body , body )
self . failUnless ( body . startswith ( " Traceback " ) , body )
2009-12-27 19:58:28 +00:00
d . addCallback ( _internal_error_text3 )
d . addCallback ( lambda ignored :
self . shouldHTTPError ( " GET errorboom_text " ,
500 , " Internal Server Error " , None ,
self . GET , " ERRORBOOM " ) )
def _internal_error_html4 ( body ) :
self . failUnless ( " <html> " in body , " expected HTML, not ' %s ' " % body )
d . addCallback ( _internal_error_html4 )
2009-03-04 04:56:30 +00:00
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 " )