2017-08-23 21:34:11 +00:00
|
|
|
import json
|
|
|
|
import shutil
|
2015-10-01 21:40:10 +00:00
|
|
|
import os.path
|
2017-08-23 21:34:11 +00:00
|
|
|
import mock
|
2015-10-01 21:40:10 +00:00
|
|
|
import re
|
2017-12-20 20:17:54 +00:00
|
|
|
import time
|
2015-10-01 21:40:10 +00:00
|
|
|
|
|
|
|
from twisted.trial import unittest
|
|
|
|
from twisted.internet import defer
|
|
|
|
from twisted.internet import reactor
|
2015-10-19 17:25:02 +00:00
|
|
|
from twisted.python import usage
|
2015-10-01 21:40:10 +00:00
|
|
|
|
2015-10-20 20:50:00 +00:00
|
|
|
from allmydata.util.assertutil import precondition
|
2015-10-01 21:40:10 +00:00
|
|
|
from allmydata.util import fileutil
|
|
|
|
from allmydata.scripts.common import get_aliases
|
2016-08-03 20:07:29 +00:00
|
|
|
from ..no_network import GridTestMixin
|
2016-12-15 05:20:37 +00:00
|
|
|
from ..common_util import parse_cli
|
2016-09-09 07:00:39 +00:00
|
|
|
from .common import CLITestMixin
|
2016-09-27 15:08:02 +00:00
|
|
|
from allmydata.test.common_util import NonASCIIPathMixin
|
2015-10-01 21:40:10 +00:00
|
|
|
from allmydata.scripts import magic_folder_cli
|
|
|
|
from allmydata.util.fileutil import abspath_expanduser_unicode
|
2015-10-20 16:30:53 +00:00
|
|
|
from allmydata.util.encodingutil import unicode_to_argv
|
2015-10-01 21:40:10 +00:00
|
|
|
from allmydata.frontends.magic_folder import MagicFolder
|
|
|
|
from allmydata import uri
|
|
|
|
|
|
|
|
|
2016-09-27 15:08:02 +00:00
|
|
|
class MagicFolderCLITestMixin(CLITestMixin, GridTestMixin, NonASCIIPathMixin):
|
|
|
|
def setUp(self):
|
|
|
|
GridTestMixin.setUp(self)
|
2016-09-27 17:05:09 +00:00
|
|
|
self.alice_nickname = self.unicode_or_fallback(u"Alice\u00F8", u"Alice", io_as_well=True)
|
|
|
|
self.bob_nickname = self.unicode_or_fallback(u"Bob\u00F8", u"Bob", io_as_well=True)
|
2016-09-27 15:08:02 +00:00
|
|
|
|
2015-10-01 21:40:10 +00:00
|
|
|
def do_create_magic_folder(self, client_num):
|
2017-08-23 21:34:11 +00:00
|
|
|
d = self.do_cli("magic-folder", "--debug", "create", "magic:", client_num=client_num)
|
2015-10-01 21:40:10 +00:00
|
|
|
def _done((rc,stdout,stderr)):
|
2016-01-28 20:33:09 +00:00
|
|
|
self.failUnlessEqual(rc, 0, stdout + stderr)
|
2015-10-01 21:40:10 +00:00
|
|
|
self.failUnlessIn("Alias 'magic' created", stdout)
|
2017-08-23 21:34:11 +00:00
|
|
|
# self.failUnlessIn("joined new magic-folder", stdout)
|
|
|
|
# self.failUnlessIn("Successfully created magic-folder", stdout)
|
2015-10-01 21:40:10 +00:00
|
|
|
self.failUnlessEqual(stderr, "")
|
|
|
|
aliases = get_aliases(self.get_clientdir(i=client_num))
|
|
|
|
self.failUnlessIn("magic", aliases)
|
|
|
|
self.failUnless(aliases["magic"].startswith("URI:DIR2:"))
|
|
|
|
d.addCallback(_done)
|
|
|
|
return d
|
|
|
|
|
|
|
|
def do_invite(self, client_num, nickname):
|
2015-10-20 16:30:53 +00:00
|
|
|
nickname_arg = unicode_to_argv(nickname)
|
|
|
|
d = self.do_cli("magic-folder", "invite", "magic:", nickname_arg, client_num=client_num)
|
2015-10-20 20:43:54 +00:00
|
|
|
def _done((rc, stdout, stderr)):
|
|
|
|
self.failUnlessEqual(rc, 0)
|
|
|
|
return (rc, stdout, stderr)
|
2015-10-01 21:40:10 +00:00
|
|
|
d.addCallback(_done)
|
|
|
|
return d
|
|
|
|
|
2017-08-23 21:34:11 +00:00
|
|
|
def do_list(self, client_num, json=False):
|
|
|
|
args = ("magic-folder", "list",)
|
|
|
|
if json:
|
|
|
|
args = args + ("--json",)
|
|
|
|
d = self.do_cli(*args, client_num=client_num)
|
|
|
|
def _done((rc, stdout, stderr)):
|
|
|
|
return (rc, stdout, stderr)
|
|
|
|
d.addCallback(_done)
|
|
|
|
return d
|
|
|
|
|
|
|
|
def do_status(self, client_num, name=None):
|
|
|
|
args = ("magic-folder", "status",)
|
|
|
|
if name is not None:
|
|
|
|
args = args + ("--name", name)
|
|
|
|
d = self.do_cli(*args, client_num=client_num)
|
|
|
|
def _done((rc, stdout, stderr)):
|
|
|
|
return (rc, stdout, stderr)
|
|
|
|
d.addCallback(_done)
|
|
|
|
return d
|
|
|
|
|
2015-10-01 21:40:10 +00:00
|
|
|
def do_join(self, client_num, local_dir, invite_code):
|
2015-10-20 20:50:00 +00:00
|
|
|
precondition(isinstance(local_dir, unicode), local_dir=local_dir)
|
|
|
|
precondition(isinstance(invite_code, str), invite_code=invite_code)
|
|
|
|
|
|
|
|
local_dir_arg = unicode_to_argv(local_dir)
|
|
|
|
d = self.do_cli("magic-folder", "join", invite_code, local_dir_arg, client_num=client_num)
|
2015-12-09 10:45:33 +00:00
|
|
|
def _done((rc, stdout, stderr)):
|
|
|
|
self.failUnlessEqual(rc, 0)
|
|
|
|
self.failUnlessEqual(stdout, "")
|
|
|
|
self.failUnlessEqual(stderr, "")
|
|
|
|
return (rc, stdout, stderr)
|
|
|
|
d.addCallback(_done)
|
|
|
|
return d
|
|
|
|
|
|
|
|
def do_leave(self, client_num):
|
|
|
|
d = self.do_cli("magic-folder", "leave", client_num=client_num)
|
2015-10-20 20:43:54 +00:00
|
|
|
def _done((rc, stdout, stderr)):
|
|
|
|
self.failUnlessEqual(rc, 0)
|
|
|
|
return (rc, stdout, stderr)
|
2015-10-01 21:40:10 +00:00
|
|
|
d.addCallback(_done)
|
|
|
|
return d
|
|
|
|
|
|
|
|
def check_joined_config(self, client_num, upload_dircap):
|
|
|
|
"""Tests that our collective directory has the readonly cap of
|
|
|
|
our upload directory.
|
|
|
|
"""
|
2017-08-23 21:34:11 +00:00
|
|
|
collective_readonly_cap = self.get_caps_from_files(client_num)[0]
|
2015-10-01 21:40:10 +00:00
|
|
|
d = self.do_cli("ls", "--json", collective_readonly_cap, client_num=client_num)
|
2015-10-20 20:43:54 +00:00
|
|
|
def _done((rc, stdout, stderr)):
|
|
|
|
self.failUnlessEqual(rc, 0)
|
|
|
|
return (rc, stdout, stderr)
|
2015-10-01 21:40:10 +00:00
|
|
|
d.addCallback(_done)
|
|
|
|
def test_joined_magic_folder((rc,stdout,stderr)):
|
|
|
|
readonly_cap = unicode(uri.from_string(upload_dircap).get_readonly().to_string(), 'utf-8')
|
|
|
|
s = re.search(readonly_cap, stdout)
|
|
|
|
self.failUnless(s is not None)
|
|
|
|
return None
|
|
|
|
d.addCallback(test_joined_magic_folder)
|
|
|
|
return d
|
|
|
|
|
|
|
|
def get_caps_from_files(self, client_num):
|
2017-08-23 21:34:11 +00:00
|
|
|
from allmydata.frontends.magic_folder import load_magic_folders
|
|
|
|
folders = load_magic_folders(self.get_clientdir(i=client_num))
|
|
|
|
mf = folders["default"]
|
|
|
|
return mf['collective_dircap'], mf['upload_dircap']
|
2015-10-01 21:40:10 +00:00
|
|
|
|
|
|
|
def check_config(self, client_num, local_dir):
|
|
|
|
client_config = fileutil.read(os.path.join(self.get_clientdir(i=client_num), "tahoe.cfg"))
|
2017-08-23 21:34:11 +00:00
|
|
|
mf_yaml = fileutil.read(os.path.join(self.get_clientdir(i=client_num), "private", "magic_folders.yaml"))
|
2015-10-20 16:31:03 +00:00
|
|
|
local_dir_utf8 = local_dir.encode('utf-8')
|
2017-08-23 21:34:11 +00:00
|
|
|
magic_folder_config = "[magic_folder]\nenabled = True"
|
2015-10-23 22:13:43 +00:00
|
|
|
self.failUnlessIn(magic_folder_config, client_config)
|
2017-08-23 21:34:11 +00:00
|
|
|
self.failUnlessIn(local_dir_utf8, mf_yaml)
|
2015-10-01 21:40:10 +00:00
|
|
|
|
|
|
|
def create_invite_join_magic_folder(self, nickname, local_dir):
|
2015-10-20 16:30:53 +00:00
|
|
|
nickname_arg = unicode_to_argv(nickname)
|
|
|
|
local_dir_arg = unicode_to_argv(local_dir)
|
2017-08-23 21:34:11 +00:00
|
|
|
# the --debug means we get real exceptions on failures
|
|
|
|
d = self.do_cli("magic-folder", "--debug", "create", "magic:", nickname_arg, local_dir_arg)
|
2015-10-20 16:27:38 +00:00
|
|
|
def _done((rc, stdout, stderr)):
|
2016-01-28 20:33:09 +00:00
|
|
|
self.failUnlessEqual(rc, 0, stdout + stderr)
|
2015-10-20 16:30:53 +00:00
|
|
|
|
2015-10-01 21:40:10 +00:00
|
|
|
client = self.get_client()
|
|
|
|
self.collective_dircap, self.upload_dircap = self.get_caps_from_files(0)
|
|
|
|
self.collective_dirnode = client.create_node_from_uri(self.collective_dircap)
|
|
|
|
self.upload_dirnode = client.create_node_from_uri(self.upload_dircap)
|
2015-10-20 16:30:53 +00:00
|
|
|
d.addCallback(_done)
|
2015-10-20 16:27:38 +00:00
|
|
|
d.addCallback(lambda ign: self.check_joined_config(0, self.upload_dircap))
|
|
|
|
d.addCallback(lambda ign: self.check_config(0, local_dir))
|
2015-10-01 21:40:10 +00:00
|
|
|
return d
|
|
|
|
|
2016-01-28 20:33:09 +00:00
|
|
|
# XXX should probably just be "tearDown"...
|
2015-10-01 21:40:10 +00:00
|
|
|
def cleanup(self, res):
|
|
|
|
d = defer.succeed(None)
|
2016-01-28 20:33:09 +00:00
|
|
|
def _clean(ign):
|
|
|
|
d = self.magicfolder.finish()
|
2016-10-24 23:09:57 +00:00
|
|
|
self.magicfolder.uploader._clock.advance(self.magicfolder.uploader._pending_delay + 1)
|
|
|
|
self.magicfolder.downloader._clock.advance(self.magicfolder.downloader._poll_interval + 1)
|
2016-01-28 20:33:09 +00:00
|
|
|
return d
|
|
|
|
|
|
|
|
d.addCallback(_clean)
|
2015-10-01 21:40:10 +00:00
|
|
|
d.addCallback(lambda ign: res)
|
|
|
|
return d
|
|
|
|
|
|
|
|
def init_magicfolder(self, client_num, upload_dircap, collective_dircap, local_magic_dir, clock):
|
2017-08-23 21:34:11 +00:00
|
|
|
dbfile = abspath_expanduser_unicode(u"magicfolder_default.sqlite", base=self.get_clientdir(i=client_num))
|
2016-08-16 22:29:37 +00:00
|
|
|
magicfolder = MagicFolder(
|
|
|
|
client=self.get_client(client_num),
|
|
|
|
upload_dircap=upload_dircap,
|
|
|
|
collective_dircap=collective_dircap,
|
|
|
|
local_path_u=local_magic_dir,
|
|
|
|
dbfile=dbfile,
|
|
|
|
umask=0o077,
|
2017-08-23 21:34:11 +00:00
|
|
|
name='default',
|
2016-08-16 22:29:37 +00:00
|
|
|
clock=clock,
|
2016-10-24 23:09:57 +00:00
|
|
|
uploader_delay=0.2,
|
|
|
|
downloader_delay=0,
|
2016-08-16 22:29:37 +00:00
|
|
|
)
|
2015-10-07 23:03:28 +00:00
|
|
|
|
2015-10-01 21:40:10 +00:00
|
|
|
magicfolder.setServiceParent(self.get_client(client_num))
|
|
|
|
magicfolder.ready()
|
|
|
|
return magicfolder
|
|
|
|
|
2015-10-07 23:03:28 +00:00
|
|
|
def setup_alice_and_bob(self, alice_clock=reactor, bob_clock=reactor):
|
2016-08-03 22:40:48 +00:00
|
|
|
self.set_up_grid(num_clients=2, oneshare=True)
|
2015-10-01 21:40:10 +00:00
|
|
|
|
2015-10-20 20:37:05 +00:00
|
|
|
self.alice_magicfolder = None
|
|
|
|
self.bob_magicfolder = None
|
|
|
|
|
2015-10-01 21:40:10 +00:00
|
|
|
alice_magic_dir = abspath_expanduser_unicode(u"Alice-magic", base=self.basedir)
|
|
|
|
self.mkdir_nonascii(alice_magic_dir)
|
|
|
|
bob_magic_dir = abspath_expanduser_unicode(u"Bob-magic", base=self.basedir)
|
|
|
|
self.mkdir_nonascii(bob_magic_dir)
|
|
|
|
|
2016-09-27 15:08:02 +00:00
|
|
|
# Alice creates a Magic Folder, invites herself and joins.
|
2015-10-01 21:40:10 +00:00
|
|
|
d = self.do_create_magic_folder(0)
|
2016-09-27 15:08:02 +00:00
|
|
|
d.addCallback(lambda ign: self.do_invite(0, self.alice_nickname))
|
2015-10-20 20:43:54 +00:00
|
|
|
def get_invite_code(result):
|
|
|
|
self.invite_code = result[1].strip()
|
|
|
|
d.addCallback(get_invite_code)
|
|
|
|
d.addCallback(lambda ign: self.do_join(0, alice_magic_dir, self.invite_code))
|
2015-10-20 16:27:38 +00:00
|
|
|
def get_alice_caps(ign):
|
2015-10-01 21:40:10 +00:00
|
|
|
self.alice_collective_dircap, self.alice_upload_dircap = self.get_caps_from_files(0)
|
|
|
|
d.addCallback(get_alice_caps)
|
2015-10-20 16:27:38 +00:00
|
|
|
d.addCallback(lambda ign: self.check_joined_config(0, self.alice_upload_dircap))
|
|
|
|
d.addCallback(lambda ign: self.check_config(0, alice_magic_dir))
|
2015-10-01 21:40:10 +00:00
|
|
|
def get_Alice_magicfolder(result):
|
2015-10-20 20:36:11 +00:00
|
|
|
self.alice_magicfolder = self.init_magicfolder(0, self.alice_upload_dircap,
|
|
|
|
self.alice_collective_dircap,
|
|
|
|
alice_magic_dir, alice_clock)
|
2015-10-01 21:40:10 +00:00
|
|
|
return result
|
|
|
|
d.addCallback(get_Alice_magicfolder)
|
|
|
|
|
|
|
|
# Alice invites Bob. Bob joins.
|
2016-09-27 15:08:02 +00:00
|
|
|
d.addCallback(lambda ign: self.do_invite(0, self.bob_nickname))
|
2015-10-20 20:43:54 +00:00
|
|
|
def get_invite_code(result):
|
|
|
|
self.invite_code = result[1].strip()
|
|
|
|
d.addCallback(get_invite_code)
|
|
|
|
d.addCallback(lambda ign: self.do_join(1, bob_magic_dir, self.invite_code))
|
2015-10-20 16:27:38 +00:00
|
|
|
def get_bob_caps(ign):
|
2015-10-01 21:40:10 +00:00
|
|
|
self.bob_collective_dircap, self.bob_upload_dircap = self.get_caps_from_files(1)
|
|
|
|
d.addCallback(get_bob_caps)
|
2015-10-20 16:27:38 +00:00
|
|
|
d.addCallback(lambda ign: self.check_joined_config(1, self.bob_upload_dircap))
|
|
|
|
d.addCallback(lambda ign: self.check_config(1, bob_magic_dir))
|
2015-10-01 21:40:10 +00:00
|
|
|
def get_Bob_magicfolder(result):
|
2015-10-20 20:36:11 +00:00
|
|
|
self.bob_magicfolder = self.init_magicfolder(1, self.bob_upload_dircap,
|
|
|
|
self.bob_collective_dircap,
|
|
|
|
bob_magic_dir, bob_clock)
|
2015-10-01 21:40:10 +00:00
|
|
|
return result
|
|
|
|
d.addCallback(get_Bob_magicfolder)
|
|
|
|
return d
|
|
|
|
|
|
|
|
|
2017-08-23 21:34:11 +00:00
|
|
|
class ListMagicFolder(MagicFolderCLITestMixin, unittest.TestCase):
|
|
|
|
|
|
|
|
@defer.inlineCallbacks
|
|
|
|
def setUp(self):
|
|
|
|
yield super(ListMagicFolder, self).setUp()
|
|
|
|
self.basedir="mf_list"
|
|
|
|
self.set_up_grid(oneshare=True)
|
|
|
|
self.local_dir = os.path.join(self.basedir, "magic")
|
|
|
|
os.mkdir(self.local_dir)
|
|
|
|
self.abs_local_dir_u = abspath_expanduser_unicode(unicode(self.local_dir), long_path=False)
|
|
|
|
|
|
|
|
yield self.do_create_magic_folder(0)
|
|
|
|
(rc, stdout, stderr) = yield self.do_invite(0, self.alice_nickname)
|
|
|
|
invite_code = stdout.strip()
|
|
|
|
yield self.do_join(0, unicode(self.local_dir), invite_code)
|
|
|
|
|
|
|
|
@defer.inlineCallbacks
|
|
|
|
def tearDown(self):
|
|
|
|
yield super(ListMagicFolder, self).tearDown()
|
|
|
|
shutil.rmtree(self.basedir)
|
|
|
|
|
|
|
|
@defer.inlineCallbacks
|
|
|
|
def test_list(self):
|
|
|
|
rc, stdout, stderr = yield self.do_list(0)
|
|
|
|
self.failUnlessEqual(rc, 0)
|
|
|
|
self.assertIn("default:", stdout)
|
|
|
|
|
|
|
|
@defer.inlineCallbacks
|
|
|
|
def test_list_none(self):
|
|
|
|
yield self.do_leave(0)
|
|
|
|
rc, stdout, stderr = yield self.do_list(0)
|
|
|
|
self.failUnlessEqual(rc, 0)
|
|
|
|
self.assertIn("No magic-folders", stdout)
|
|
|
|
|
|
|
|
@defer.inlineCallbacks
|
|
|
|
def test_list_json(self):
|
|
|
|
rc, stdout, stderr = yield self.do_list(0, json=True)
|
|
|
|
self.failUnlessEqual(rc, 0)
|
|
|
|
res = json.loads(stdout)
|
|
|
|
self.assertEqual(
|
|
|
|
dict(default=dict(directory=self.abs_local_dir_u)),
|
|
|
|
res,
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
class StatusMagicFolder(MagicFolderCLITestMixin, unittest.TestCase):
|
|
|
|
|
|
|
|
@defer.inlineCallbacks
|
|
|
|
def setUp(self):
|
|
|
|
yield super(StatusMagicFolder, self).setUp()
|
|
|
|
self.basedir="mf_list"
|
|
|
|
self.set_up_grid(oneshare=True)
|
|
|
|
self.local_dir = os.path.join(self.basedir, "magic")
|
|
|
|
os.mkdir(self.local_dir)
|
|
|
|
self.abs_local_dir_u = abspath_expanduser_unicode(unicode(self.local_dir), long_path=False)
|
|
|
|
|
|
|
|
yield self.do_create_magic_folder(0)
|
|
|
|
(rc, stdout, stderr) = yield self.do_invite(0, self.alice_nickname)
|
|
|
|
invite_code = stdout.strip()
|
|
|
|
yield self.do_join(0, unicode(self.local_dir), invite_code)
|
|
|
|
|
|
|
|
@defer.inlineCallbacks
|
|
|
|
def tearDown(self):
|
|
|
|
yield super(StatusMagicFolder, self).tearDown()
|
|
|
|
shutil.rmtree(self.basedir)
|
|
|
|
|
|
|
|
@defer.inlineCallbacks
|
|
|
|
def test_status(self):
|
2017-12-20 20:17:54 +00:00
|
|
|
secs_per_year = 365 * 24 * 60 * 60.0
|
|
|
|
|
2017-08-23 21:34:11 +00:00
|
|
|
def json_for_cap(options, cap):
|
|
|
|
if cap.startswith('URI:DIR2:'):
|
|
|
|
return (
|
|
|
|
'dirnode',
|
|
|
|
{
|
|
|
|
"children": {
|
|
|
|
"foo": ('filenode', {
|
|
|
|
"size": 1234,
|
|
|
|
"metadata": {
|
|
|
|
"tahoe": {
|
2017-12-20 20:17:54 +00:00
|
|
|
"linkcrtime": (time.time() - (5 * secs_per_year)),
|
2017-08-23 21:34:11 +00:00
|
|
|
},
|
|
|
|
"version": 1,
|
|
|
|
},
|
|
|
|
"ro_uri": "read-only URI",
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)
|
|
|
|
else:
|
|
|
|
return ('dirnode', {"children": {}})
|
|
|
|
jc = mock.patch(
|
|
|
|
"allmydata.scripts.magic_folder_cli._get_json_for_cap",
|
|
|
|
side_effect=json_for_cap,
|
|
|
|
)
|
|
|
|
|
|
|
|
def json_for_frag(options, fragment, method='GET', post_args=None):
|
|
|
|
return {}
|
|
|
|
jf = mock.patch(
|
|
|
|
"allmydata.scripts.magic_folder_cli._get_json_for_fragment",
|
|
|
|
side_effect=json_for_frag,
|
|
|
|
)
|
|
|
|
|
|
|
|
with jc, jf:
|
|
|
|
rc, stdout, stderr = yield self.do_status(0)
|
|
|
|
self.failUnlessEqual(rc, 0)
|
|
|
|
self.assertIn("default", stdout)
|
|
|
|
|
|
|
|
self.assertIn(
|
2017-12-20 20:17:54 +00:00
|
|
|
"foo (1.23 kB): good, version=1, created 5 years ago",
|
2017-08-23 21:34:11 +00:00
|
|
|
stdout,
|
|
|
|
)
|
|
|
|
|
|
|
|
@defer.inlineCallbacks
|
|
|
|
def test_status_child_not_dirnode(self):
|
|
|
|
def json_for_cap(options, cap):
|
|
|
|
if cap.startswith('URI:DIR2'):
|
|
|
|
return (
|
|
|
|
'dirnode',
|
|
|
|
{
|
|
|
|
"children": {
|
|
|
|
"foo": ('filenode', {
|
|
|
|
"size": 1234,
|
|
|
|
"metadata": {
|
|
|
|
"tahoe": {
|
|
|
|
"linkcrtime": 0.0,
|
|
|
|
},
|
|
|
|
"version": 1,
|
|
|
|
},
|
|
|
|
"ro_uri": "read-only URI",
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)
|
|
|
|
elif cap == "read-only URI":
|
|
|
|
return {
|
|
|
|
"error": "bad stuff",
|
|
|
|
}
|
|
|
|
else:
|
|
|
|
return ('dirnode', {"children": {}})
|
|
|
|
jc = mock.patch(
|
|
|
|
"allmydata.scripts.magic_folder_cli._get_json_for_cap",
|
|
|
|
side_effect=json_for_cap,
|
|
|
|
)
|
|
|
|
|
|
|
|
def json_for_frag(options, fragment, method='GET', post_args=None):
|
|
|
|
return {}
|
|
|
|
jf = mock.patch(
|
|
|
|
"allmydata.scripts.magic_folder_cli._get_json_for_fragment",
|
|
|
|
side_effect=json_for_frag,
|
|
|
|
)
|
|
|
|
|
|
|
|
with jc, jf:
|
|
|
|
rc, stdout, stderr = yield self.do_status(0)
|
|
|
|
self.failUnlessEqual(rc, 0)
|
|
|
|
|
|
|
|
self.assertIn(
|
|
|
|
"expected a dirnode",
|
|
|
|
stdout + stderr,
|
|
|
|
)
|
|
|
|
|
|
|
|
@defer.inlineCallbacks
|
|
|
|
def test_status_error_not_dircap(self):
|
|
|
|
def json_for_cap(options, cap):
|
|
|
|
if cap.startswith('URI:DIR2:'):
|
|
|
|
return (
|
|
|
|
'filenode',
|
|
|
|
{}
|
|
|
|
)
|
|
|
|
else:
|
|
|
|
return ('dirnode', {"children": {}})
|
|
|
|
jc = mock.patch(
|
|
|
|
"allmydata.scripts.magic_folder_cli._get_json_for_cap",
|
|
|
|
side_effect=json_for_cap,
|
|
|
|
)
|
|
|
|
|
|
|
|
def json_for_frag(options, fragment, method='GET', post_args=None):
|
|
|
|
return {}
|
|
|
|
jf = mock.patch(
|
|
|
|
"allmydata.scripts.magic_folder_cli._get_json_for_fragment",
|
|
|
|
side_effect=json_for_frag,
|
|
|
|
)
|
|
|
|
|
|
|
|
with jc, jf:
|
|
|
|
rc, stdout, stderr = yield self.do_status(0)
|
|
|
|
self.failUnlessEqual(rc, 2)
|
|
|
|
self.assertIn(
|
|
|
|
"magic_folder_dircap isn't a directory capability",
|
|
|
|
stdout + stderr,
|
|
|
|
)
|
|
|
|
|
|
|
|
@defer.inlineCallbacks
|
|
|
|
def test_status_nothing(self):
|
|
|
|
rc, stdout, stderr = yield self.do_status(0, name="blam")
|
|
|
|
self.assertIn("No such magic-folder 'blam'", stderr)
|
|
|
|
|
|
|
|
|
2015-10-01 21:40:10 +00:00
|
|
|
class CreateMagicFolder(MagicFolderCLITestMixin, unittest.TestCase):
|
|
|
|
def test_create_and_then_invite_join(self):
|
|
|
|
self.basedir = "cli/MagicFolder/create-and-then-invite-join"
|
2016-08-03 22:40:48 +00:00
|
|
|
self.set_up_grid(oneshare=True)
|
2015-10-20 16:30:53 +00:00
|
|
|
local_dir = os.path.join(self.basedir, "magic")
|
2017-08-23 21:34:11 +00:00
|
|
|
os.mkdir(local_dir)
|
2015-10-23 21:04:08 +00:00
|
|
|
abs_local_dir_u = abspath_expanduser_unicode(unicode(local_dir), long_path=False)
|
2015-10-20 16:30:53 +00:00
|
|
|
|
2015-10-01 21:40:10 +00:00
|
|
|
d = self.do_create_magic_folder(0)
|
2016-09-27 15:08:02 +00:00
|
|
|
d.addCallback(lambda ign: self.do_invite(0, self.alice_nickname))
|
2015-10-20 16:30:53 +00:00
|
|
|
def get_invite_code_and_join((rc, stdout, stderr)):
|
|
|
|
invite_code = stdout.strip()
|
2015-10-20 20:50:00 +00:00
|
|
|
return self.do_join(0, unicode(local_dir), invite_code)
|
2015-10-20 16:30:53 +00:00
|
|
|
d.addCallback(get_invite_code_and_join)
|
2015-10-20 16:27:38 +00:00
|
|
|
def get_caps(ign):
|
2015-10-01 21:40:10 +00:00
|
|
|
self.collective_dircap, self.upload_dircap = self.get_caps_from_files(0)
|
|
|
|
d.addCallback(get_caps)
|
2015-10-20 16:27:38 +00:00
|
|
|
d.addCallback(lambda ign: self.check_joined_config(0, self.upload_dircap))
|
2015-10-23 21:04:08 +00:00
|
|
|
d.addCallback(lambda ign: self.check_config(0, abs_local_dir_u))
|
2015-10-01 21:40:10 +00:00
|
|
|
return d
|
|
|
|
|
|
|
|
def test_create_error(self):
|
|
|
|
self.basedir = "cli/MagicFolder/create-error"
|
2016-08-03 22:40:48 +00:00
|
|
|
self.set_up_grid(oneshare=True)
|
2015-10-20 16:30:53 +00:00
|
|
|
|
2015-10-01 21:40:10 +00:00
|
|
|
d = self.do_cli("magic-folder", "create", "m a g i c:", client_num=0)
|
2015-10-20 16:27:38 +00:00
|
|
|
def _done((rc, stdout, stderr)):
|
2015-10-01 21:40:10 +00:00
|
|
|
self.failIfEqual(rc, 0)
|
|
|
|
self.failUnlessIn("Alias names cannot contain spaces.", stderr)
|
|
|
|
d.addCallback(_done)
|
|
|
|
return d
|
|
|
|
|
2017-08-23 21:34:11 +00:00
|
|
|
@defer.inlineCallbacks
|
|
|
|
def test_create_duplicate_name(self):
|
|
|
|
self.basedir = "cli/MagicFolder/create-dup"
|
|
|
|
self.set_up_grid(oneshare=True)
|
|
|
|
|
|
|
|
rc, stdout, stderr = yield self.do_cli(
|
|
|
|
"magic-folder", "create", "magic:", "--name", "foo",
|
|
|
|
client_num=0,
|
|
|
|
)
|
|
|
|
self.assertEqual(rc, 0)
|
|
|
|
|
|
|
|
rc, stdout, stderr = yield self.do_cli(
|
|
|
|
"magic-folder", "create", "magic:", "--name", "foo",
|
|
|
|
client_num=0,
|
|
|
|
)
|
|
|
|
self.assertEqual(rc, 1)
|
|
|
|
self.assertIn(
|
|
|
|
"Already have a magic-folder named 'default'",
|
|
|
|
stderr
|
|
|
|
)
|
|
|
|
|
|
|
|
@defer.inlineCallbacks
|
|
|
|
def test_leave_wrong_folder(self):
|
|
|
|
self.basedir = "cli/MagicFolder/leave_wrong_folders"
|
|
|
|
yield self.set_up_grid(oneshare=True)
|
|
|
|
magic_dir = os.path.join(self.basedir, 'magic')
|
|
|
|
os.mkdir(magic_dir)
|
|
|
|
|
|
|
|
rc, stdout, stderr = yield self.do_cli(
|
|
|
|
"magic-folder", "create", "--name", "foo", "magic:", "my_name", magic_dir,
|
|
|
|
client_num=0,
|
|
|
|
)
|
|
|
|
self.assertEqual(rc, 0)
|
|
|
|
|
|
|
|
rc, stdout, stderr = yield self.do_cli(
|
|
|
|
"magic-folder", "leave", "--name", "bar",
|
|
|
|
client_num=0,
|
|
|
|
)
|
|
|
|
self.assertNotEqual(rc, 0)
|
|
|
|
self.assertIn(
|
|
|
|
"No such magic-folder 'bar'",
|
|
|
|
stdout + stderr,
|
|
|
|
)
|
|
|
|
|
|
|
|
@defer.inlineCallbacks
|
|
|
|
def test_leave_no_folder(self):
|
|
|
|
self.basedir = "cli/MagicFolder/leave_no_folders"
|
|
|
|
yield self.set_up_grid(oneshare=True)
|
|
|
|
magic_dir = os.path.join(self.basedir, 'magic')
|
|
|
|
os.mkdir(magic_dir)
|
|
|
|
|
|
|
|
rc, stdout, stderr = yield self.do_cli(
|
|
|
|
"magic-folder", "create", "--name", "foo", "magic:", "my_name", magic_dir,
|
|
|
|
client_num=0,
|
|
|
|
)
|
|
|
|
self.assertEqual(rc, 0)
|
|
|
|
|
|
|
|
rc, stdout, stderr = yield self.do_cli(
|
|
|
|
"magic-folder", "leave", "--name", "foo",
|
|
|
|
client_num=0,
|
|
|
|
)
|
|
|
|
self.assertEqual(rc, 0)
|
|
|
|
|
|
|
|
rc, stdout, stderr = yield self.do_cli(
|
|
|
|
"magic-folder", "leave", "--name", "foo",
|
|
|
|
client_num=0,
|
|
|
|
)
|
|
|
|
self.assertEqual(rc, 1)
|
|
|
|
self.assertIn(
|
|
|
|
"No magic-folders at all",
|
|
|
|
stderr,
|
|
|
|
)
|
|
|
|
|
|
|
|
@defer.inlineCallbacks
|
|
|
|
def test_leave_no_folders_at_all(self):
|
|
|
|
self.basedir = "cli/MagicFolder/leave_no_folders_at_all"
|
|
|
|
yield self.set_up_grid(oneshare=True)
|
|
|
|
|
|
|
|
rc, stdout, stderr = yield self.do_cli(
|
|
|
|
"magic-folder", "leave",
|
|
|
|
client_num=0,
|
|
|
|
)
|
|
|
|
self.assertEqual(rc, 1)
|
|
|
|
self.assertIn(
|
|
|
|
"No magic-folders at all",
|
|
|
|
stderr,
|
|
|
|
)
|
|
|
|
|
2015-10-01 21:40:10 +00:00
|
|
|
def test_create_invite_join(self):
|
|
|
|
self.basedir = "cli/MagicFolder/create-invite-join"
|
2016-08-03 22:40:48 +00:00
|
|
|
self.set_up_grid(oneshare=True)
|
2015-10-20 16:30:53 +00:00
|
|
|
local_dir = os.path.join(self.basedir, "magic")
|
2015-10-23 21:04:08 +00:00
|
|
|
abs_local_dir_u = abspath_expanduser_unicode(unicode(local_dir), long_path=False)
|
2015-10-20 16:30:53 +00:00
|
|
|
|
|
|
|
d = self.do_cli("magic-folder", "create", "magic:", "Alice", local_dir)
|
2015-10-20 16:27:38 +00:00
|
|
|
def _done((rc, stdout, stderr)):
|
2015-10-20 20:43:54 +00:00
|
|
|
self.failUnlessEqual(rc, 0)
|
2015-10-01 21:40:10 +00:00
|
|
|
self.collective_dircap, self.upload_dircap = self.get_caps_from_files(0)
|
2015-10-20 20:43:54 +00:00
|
|
|
d.addCallback(_done)
|
2015-10-20 16:27:38 +00:00
|
|
|
d.addCallback(lambda ign: self.check_joined_config(0, self.upload_dircap))
|
2015-10-23 21:04:08 +00:00
|
|
|
d.addCallback(lambda ign: self.check_config(0, abs_local_dir_u))
|
2015-10-01 21:40:10 +00:00
|
|
|
return d
|
2015-10-19 14:02:28 +00:00
|
|
|
|
2017-11-28 20:36:49 +00:00
|
|
|
def test_help_synopsis(self):
|
|
|
|
self.basedir = "cli/MagicFolder/help_synopsis"
|
|
|
|
os.makedirs(self.basedir)
|
|
|
|
|
|
|
|
o = magic_folder_cli.CreateOptions()
|
|
|
|
o.parent = magic_folder_cli.MagicFolderCommand()
|
|
|
|
o.parent.getSynopsis()
|
|
|
|
|
2015-10-19 14:02:28 +00:00
|
|
|
def test_create_invite_join_failure(self):
|
2015-10-20 11:08:14 +00:00
|
|
|
self.basedir = "cli/MagicFolder/create-invite-join-failure"
|
2015-10-20 16:31:03 +00:00
|
|
|
os.makedirs(self.basedir)
|
|
|
|
|
2015-10-19 17:25:02 +00:00
|
|
|
o = magic_folder_cli.CreateOptions()
|
|
|
|
o.parent = magic_folder_cli.MagicFolderCommand()
|
2015-10-20 16:31:03 +00:00
|
|
|
o.parent['node-directory'] = self.basedir
|
2015-10-19 17:25:02 +00:00
|
|
|
try:
|
2015-10-20 11:08:14 +00:00
|
|
|
o.parseArgs("magic:", "Alice", "-foo")
|
2015-10-19 17:25:02 +00:00
|
|
|
except usage.UsageError as e:
|
|
|
|
self.failUnlessIn("cannot start with '-'", str(e))
|
|
|
|
else:
|
|
|
|
self.fail("expected UsageError")
|
|
|
|
|
|
|
|
def test_join_failure(self):
|
2015-10-20 11:08:14 +00:00
|
|
|
self.basedir = "cli/MagicFolder/create-join-failure"
|
2015-10-20 16:31:03 +00:00
|
|
|
os.makedirs(self.basedir)
|
2015-10-20 11:08:14 +00:00
|
|
|
|
2015-10-19 17:25:02 +00:00
|
|
|
o = magic_folder_cli.JoinOptions()
|
|
|
|
o.parent = magic_folder_cli.MagicFolderCommand()
|
2015-10-20 16:31:03 +00:00
|
|
|
o.parent['node-directory'] = self.basedir
|
2015-10-19 17:25:02 +00:00
|
|
|
try:
|
2015-10-20 11:08:14 +00:00
|
|
|
o.parseArgs("URI:invite+URI:code", "-foo")
|
2015-10-19 17:25:02 +00:00
|
|
|
except usage.UsageError as e:
|
|
|
|
self.failUnlessIn("cannot start with '-'", str(e))
|
|
|
|
else:
|
|
|
|
self.fail("expected UsageError")
|
2015-12-03 02:44:54 +00:00
|
|
|
|
|
|
|
def test_join_twice_failure(self):
|
|
|
|
self.basedir = "cli/MagicFolder/create-join-twice-failure"
|
|
|
|
os.makedirs(self.basedir)
|
2016-08-03 22:40:48 +00:00
|
|
|
self.set_up_grid(oneshare=True)
|
2015-12-03 02:44:54 +00:00
|
|
|
local_dir = os.path.join(self.basedir, "magic")
|
|
|
|
abs_local_dir_u = abspath_expanduser_unicode(unicode(local_dir), long_path=False)
|
|
|
|
|
|
|
|
d = self.do_create_magic_folder(0)
|
2016-09-27 15:08:02 +00:00
|
|
|
d.addCallback(lambda ign: self.do_invite(0, self.alice_nickname))
|
2015-12-03 02:44:54 +00:00
|
|
|
def get_invite_code_and_join((rc, stdout, stderr)):
|
|
|
|
self.invite_code = stdout.strip()
|
|
|
|
return self.do_join(0, unicode(local_dir), self.invite_code)
|
|
|
|
d.addCallback(get_invite_code_and_join)
|
|
|
|
def get_caps(ign):
|
|
|
|
self.collective_dircap, self.upload_dircap = self.get_caps_from_files(0)
|
|
|
|
d.addCallback(get_caps)
|
|
|
|
d.addCallback(lambda ign: self.check_joined_config(0, self.upload_dircap))
|
|
|
|
d.addCallback(lambda ign: self.check_config(0, abs_local_dir_u))
|
|
|
|
def join_again(ignore):
|
|
|
|
return self.do_cli("magic-folder", "join", self.invite_code, local_dir, client_num=0)
|
|
|
|
d.addCallback(join_again)
|
|
|
|
def get_results(result):
|
2015-12-21 21:30:29 +00:00
|
|
|
(rc, out, err) = result
|
|
|
|
self.failUnlessEqual(out, "")
|
2017-08-23 21:34:11 +00:00
|
|
|
self.failUnlessIn("This client already has a magic-folder", err)
|
2015-12-21 21:30:29 +00:00
|
|
|
self.failIfEqual(rc, 0)
|
2015-12-03 02:44:54 +00:00
|
|
|
d.addCallback(get_results)
|
|
|
|
return d
|
2015-12-03 12:58:31 +00:00
|
|
|
|
|
|
|
def test_join_leave_join(self):
|
|
|
|
self.basedir = "cli/MagicFolder/create-join-leave-join"
|
|
|
|
os.makedirs(self.basedir)
|
2016-08-03 22:40:48 +00:00
|
|
|
self.set_up_grid(oneshare=True)
|
2015-12-03 12:58:31 +00:00
|
|
|
local_dir = os.path.join(self.basedir, "magic")
|
|
|
|
abs_local_dir_u = abspath_expanduser_unicode(unicode(local_dir), long_path=False)
|
|
|
|
|
|
|
|
self.invite_code = None
|
|
|
|
d = self.do_create_magic_folder(0)
|
2016-09-27 15:08:02 +00:00
|
|
|
d.addCallback(lambda ign: self.do_invite(0, self.alice_nickname))
|
2015-12-03 12:58:31 +00:00
|
|
|
def get_invite_code_and_join((rc, stdout, stderr)):
|
2015-12-21 21:30:29 +00:00
|
|
|
self.failUnlessEqual(rc, 0)
|
2015-12-03 12:58:31 +00:00
|
|
|
self.invite_code = stdout.strip()
|
|
|
|
return self.do_join(0, unicode(local_dir), self.invite_code)
|
|
|
|
d.addCallback(get_invite_code_and_join)
|
|
|
|
def get_caps(ign):
|
|
|
|
self.collective_dircap, self.upload_dircap = self.get_caps_from_files(0)
|
|
|
|
d.addCallback(get_caps)
|
|
|
|
d.addCallback(lambda ign: self.check_joined_config(0, self.upload_dircap))
|
|
|
|
d.addCallback(lambda ign: self.check_config(0, abs_local_dir_u))
|
2015-12-09 10:45:33 +00:00
|
|
|
d.addCallback(lambda ign: self.do_leave(0))
|
2015-12-03 12:58:31 +00:00
|
|
|
|
2015-12-21 21:30:29 +00:00
|
|
|
d.addCallback(lambda ign: self.do_join(0, unicode(local_dir), self.invite_code))
|
2015-12-03 12:58:31 +00:00
|
|
|
def get_caps(ign):
|
|
|
|
self.collective_dircap, self.upload_dircap = self.get_caps_from_files(0)
|
|
|
|
d.addCallback(get_caps)
|
|
|
|
d.addCallback(lambda ign: self.check_joined_config(0, self.upload_dircap))
|
|
|
|
d.addCallback(lambda ign: self.check_config(0, abs_local_dir_u))
|
|
|
|
|
|
|
|
return d
|
|
|
|
|
|
|
|
def test_join_failures(self):
|
|
|
|
self.basedir = "cli/MagicFolder/create-join-failures"
|
|
|
|
os.makedirs(self.basedir)
|
2016-08-03 22:40:48 +00:00
|
|
|
self.set_up_grid(oneshare=True)
|
2015-12-03 12:58:31 +00:00
|
|
|
local_dir = os.path.join(self.basedir, "magic")
|
2017-08-23 21:34:11 +00:00
|
|
|
os.mkdir(local_dir)
|
2015-12-03 12:58:31 +00:00
|
|
|
abs_local_dir_u = abspath_expanduser_unicode(unicode(local_dir), long_path=False)
|
|
|
|
|
|
|
|
self.invite_code = None
|
|
|
|
d = self.do_create_magic_folder(0)
|
2016-09-27 15:08:02 +00:00
|
|
|
d.addCallback(lambda ign: self.do_invite(0, self.alice_nickname))
|
2015-12-03 12:58:31 +00:00
|
|
|
def get_invite_code_and_join((rc, stdout, stderr)):
|
2015-12-21 21:30:29 +00:00
|
|
|
self.failUnlessEqual(rc, 0)
|
2015-12-03 12:58:31 +00:00
|
|
|
self.invite_code = stdout.strip()
|
|
|
|
return self.do_join(0, unicode(local_dir), self.invite_code)
|
|
|
|
d.addCallback(get_invite_code_and_join)
|
|
|
|
def get_caps(ign):
|
|
|
|
self.collective_dircap, self.upload_dircap = self.get_caps_from_files(0)
|
|
|
|
d.addCallback(get_caps)
|
|
|
|
d.addCallback(lambda ign: self.check_joined_config(0, self.upload_dircap))
|
|
|
|
d.addCallback(lambda ign: self.check_config(0, abs_local_dir_u))
|
|
|
|
|
2015-12-21 21:30:29 +00:00
|
|
|
def check_success(result):
|
|
|
|
(rc, out, err) = result
|
2017-08-23 21:34:11 +00:00
|
|
|
self.failUnlessEqual(rc, 0, out + err)
|
2015-12-21 21:30:29 +00:00
|
|
|
def check_failure(result):
|
|
|
|
(rc, out, err) = result
|
|
|
|
self.failIfEqual(rc, 0)
|
|
|
|
|
|
|
|
def leave(ign):
|
2015-12-03 12:58:31 +00:00
|
|
|
return self.do_cli("magic-folder", "leave", client_num=0)
|
|
|
|
d.addCallback(leave)
|
2015-12-21 21:30:29 +00:00
|
|
|
d.addCallback(check_success)
|
2015-12-03 12:58:31 +00:00
|
|
|
|
2017-08-23 21:34:11 +00:00
|
|
|
magic_folder_db_file = os.path.join(self.get_clientdir(i=0), u"private", u"magicfolder_default.sqlite")
|
2015-12-03 12:58:31 +00:00
|
|
|
|
|
|
|
def check_join_if_file(my_file):
|
|
|
|
fileutil.write(my_file, "my file data")
|
2015-12-21 21:30:29 +00:00
|
|
|
d2 = self.do_cli("magic-folder", "join", self.invite_code, local_dir, client_num=0)
|
|
|
|
d2.addCallback(check_failure)
|
2015-12-03 12:58:31 +00:00
|
|
|
return d2
|
|
|
|
|
2017-08-23 21:34:11 +00:00
|
|
|
for my_file in [magic_folder_db_file]:
|
2015-12-03 12:58:31 +00:00
|
|
|
d.addCallback(lambda ign, my_file: check_join_if_file(my_file), my_file)
|
|
|
|
d.addCallback(leave)
|
2017-08-23 21:34:11 +00:00
|
|
|
# we didn't successfully join, so leaving should be an error
|
|
|
|
d.addCallback(check_failure)
|
2015-12-03 12:58:31 +00:00
|
|
|
|
|
|
|
return d
|
2016-12-15 05:20:37 +00:00
|
|
|
|
|
|
|
class CreateErrors(unittest.TestCase):
|
|
|
|
def test_poll_interval(self):
|
|
|
|
e = self.assertRaises(usage.UsageError, parse_cli,
|
|
|
|
"magic-folder", "create", "--poll-interval=frog",
|
|
|
|
"alias:")
|
|
|
|
self.assertEqual(str(e), "--poll-interval must be a positive integer")
|
|
|
|
|
|
|
|
e = self.assertRaises(usage.UsageError, parse_cli,
|
|
|
|
"magic-folder", "create", "--poll-interval=-4",
|
|
|
|
"alias:")
|
|
|
|
self.assertEqual(str(e), "--poll-interval must be a positive integer")
|
|
|
|
|
|
|
|
def test_alias(self):
|
|
|
|
e = self.assertRaises(usage.UsageError, parse_cli,
|
|
|
|
"magic-folder", "create", "no-colon")
|
|
|
|
self.assertEqual(str(e), "An alias must end with a ':' character.")
|
|
|
|
|
|
|
|
def test_nickname(self):
|
|
|
|
e = self.assertRaises(usage.UsageError, parse_cli,
|
|
|
|
"magic-folder", "create", "alias:", "nickname")
|
|
|
|
self.assertEqual(str(e), "If NICKNAME is specified then LOCAL_DIR must also be specified.")
|
|
|
|
|
|
|
|
class InviteErrors(unittest.TestCase):
|
|
|
|
def test_alias(self):
|
|
|
|
e = self.assertRaises(usage.UsageError, parse_cli,
|
|
|
|
"magic-folder", "invite", "no-colon")
|
|
|
|
self.assertEqual(str(e), "An alias must end with a ':' character.")
|
|
|
|
|
|
|
|
class JoinErrors(unittest.TestCase):
|
|
|
|
def test_poll_interval(self):
|
|
|
|
e = self.assertRaises(usage.UsageError, parse_cli,
|
|
|
|
"magic-folder", "join", "--poll-interval=frog",
|
|
|
|
"code", "localdir")
|
|
|
|
self.assertEqual(str(e), "--poll-interval must be a positive integer")
|
|
|
|
|
|
|
|
e = self.assertRaises(usage.UsageError, parse_cli,
|
|
|
|
"magic-folder", "join", "--poll-interval=-2",
|
|
|
|
"code", "localdir")
|
|
|
|
self.assertEqual(str(e), "--poll-interval must be a positive integer")
|