From 0737a247126ec1df97e7159c2778762783524ce6 Mon Sep 17 00:00:00 2001 From: Atul Varma Date: Tue, 23 May 2017 20:36:34 -0400 Subject: [PATCH 1/4] Add --ignore-unsendable-files flag. --- src/wormhole/cli/cli.py | 8 ++++++-- src/wormhole/cli/cmd_send.py | 16 ++++++++++++---- src/wormhole/errors.py | 10 ++++++++++ 3 files changed, 28 insertions(+), 6 deletions(-) diff --git a/src/wormhole/cli/cli.py b/src/wormhole/cli/cli.py index ba5ec2c..46b6d68 100644 --- a/src/wormhole/cli/cli.py +++ b/src/wormhole/cli/cli.py @@ -10,7 +10,7 @@ from . import public_relay from .. import __version__ from ..timing import DebugTiming from ..errors import (WrongPasswordError, WelcomeError, KeyFormatError, - TransferError, NoTorError) + TransferError, NoTorError, UnsendableFileError) from twisted.internet.defer import inlineCallbacks, maybeDeferred from twisted.python.failure import Failure from twisted.internet.task import react @@ -109,7 +109,7 @@ def _dispatch_command(reactor, cfg, command): msg = fill("ERROR: " + dedent(e.__doc__)) print(msg, file=cfg.stderr) raise SystemExit(1) - except WelcomeError as e: + except (WelcomeError, UnsendableFileError) as e: msg = fill("ERROR: " + dedent(e.__doc__)) print(msg, file=cfg.stderr) print(six.u(""), file=cfg.stderr) @@ -173,6 +173,10 @@ TorArgs = _compose( "--text", default=None, metavar="MESSAGE", help="text message to send, instead of a file. Use '-' to read from stdin.", ) +@click.option( + "--ignore-unsendable-files", default=False, is_flag=True, + help="Don't raise an error if a file can't be read." +) @click.argument("what", required=False) @click.pass_obj def send(cfg, **kwargs): diff --git a/src/wormhole/cli/cmd_send.py b/src/wormhole/cli/cmd_send.py index 307494b..4c96d9b 100644 --- a/src/wormhole/cli/cmd_send.py +++ b/src/wormhole/cli/cmd_send.py @@ -6,7 +6,8 @@ from twisted.python import log from twisted.protocols import basic from twisted.internet import reactor from twisted.internet.defer import inlineCallbacks, returnValue -from ..errors import TransferError, WormholeClosedError, NoTorError +from ..errors import (TransferError, WormholeClosedError, NoTorError, + UnsendableFileError) from wormhole import create, __version__ from ..transit import TransitSender from ..util import dict_to_bytes, bytes_to_dict, bytes_to_hexstr @@ -274,9 +275,16 @@ class Sender: for fn in files: archivename = os.path.join(*tuple(localpath+[fn])) localfilename = os.path.join(path, fn) - zf.write(localfilename, archivename) - num_bytes += os.stat(localfilename).st_size - num_files += 1 + try: + zf.write(localfilename, archivename) + num_bytes += os.stat(localfilename).st_size + num_files += 1 + except OSError as e: + errmsg = u"{}: {}".format(fn, e.strerror) + if self._args.ignore_unsendable_files: + print(u"{} (ignoring error)".format(errmsg)) + else: + raise UnsendableFileError(errmsg) fd_to_send.seek(0,2) filesize = fd_to_send.tell() fd_to_send.seek(0,0) diff --git a/src/wormhole/errors.py b/src/wormhole/errors.py index 06f74d3..7763f73 100644 --- a/src/wormhole/errors.py +++ b/src/wormhole/errors.py @@ -3,6 +3,16 @@ from __future__ import unicode_literals class WormholeError(Exception): """Parent class for all wormhole-related errors""" +class UnsendableFileError(Exception): + """ + A file you wanted to send couldn't be read, maybe because it's not + a file, or because it's a symlink that points to something + that doesn't exist. + + To ignore this kind of error, you can run wormhole with the + --ignore-unsendable-files flag. + """ + class ServerError(WormholeError): """The relay server complained about something we did.""" From 067b47fceb9f5dde4d05eb436cbbdd5d56aa42ca Mon Sep 17 00:00:00 2001 From: Atul Varma Date: Tue, 23 May 2017 21:01:40 -0400 Subject: [PATCH 2/4] Add test_broken_symlink. --- src/wormhole/test/test_cli.py | 21 ++++++++++++++++++++- 1 file changed, 20 insertions(+), 1 deletion(-) diff --git a/src/wormhole/test/test_cli.py b/src/wormhole/test/test_cli.py index 3dd23c7..c6ede48 100644 --- a/src/wormhole/test/test_cli.py +++ b/src/wormhole/test/test_cli.py @@ -12,7 +12,8 @@ from twisted.internet.defer import gatherResults, inlineCallbacks, returnValue from .. import __version__ from .common import ServerBase, config from ..cli import cmd_send, cmd_receive, welcome, cli -from ..errors import TransferError, WrongPasswordError, WelcomeError +from ..errors import (TransferError, WrongPasswordError, WelcomeError, + UnsendableFileError) from wormhole.server.cmd_server import MyPlugin from wormhole.server.cli import server @@ -65,6 +66,24 @@ class OfferData(unittest.TestCase): self.assertEqual(fd_to_send.tell(), 0) self.assertEqual(fd_to_send.read(), message) + def test_broken_symlink(self): + if not hasattr(os, 'symlink'): + raise unittest.SkipTest("host OS does not support symlinks") + + parent_dir = self.mktemp() + os.mkdir(parent_dir) + send_dir = "dirname" + os.mkdir(os.path.join(parent_dir, send_dir)) + os.symlink('/non/existent/file', + os.path.join(parent_dir, send_dir, 'linky')) + + send_dir_arg = send_dir + self.cfg.what = send_dir_arg + self.cfg.cwd = parent_dir + + e = self.assertRaises(UnsendableFileError, build_offer, self.cfg) + self.assertEqual(str(e), "linky: No such file or directory") + def test_missing_file(self): self.cfg.what = filename = "missing" send_dir = self.mktemp() From ecfd46d16d9077f1ec92277bbeb03e4e40d05a14 Mon Sep 17 00:00:00 2001 From: Atul Varma Date: Tue, 23 May 2017 21:51:40 -0400 Subject: [PATCH 3/4] Add test_broken_symlink_is_ignored. --- src/wormhole/cli/cmd_send.py | 3 ++- src/wormhole/test/test_cli.py | 13 ++++++++++++- 2 files changed, 14 insertions(+), 2 deletions(-) diff --git a/src/wormhole/cli/cmd_send.py b/src/wormhole/cli/cmd_send.py index 4c96d9b..aaa274a 100644 --- a/src/wormhole/cli/cmd_send.py +++ b/src/wormhole/cli/cmd_send.py @@ -282,7 +282,8 @@ class Sender: except OSError as e: errmsg = u"{}: {}".format(fn, e.strerror) if self._args.ignore_unsendable_files: - print(u"{} (ignoring error)".format(errmsg)) + print(u"{} (ignoring error)".format(errmsg), + file=args.stderr) else: raise UnsendableFileError(errmsg) fd_to_send.seek(0,2) diff --git a/src/wormhole/test/test_cli.py b/src/wormhole/test/test_cli.py index c6ede48..c0eca38 100644 --- a/src/wormhole/test/test_cli.py +++ b/src/wormhole/test/test_cli.py @@ -66,7 +66,7 @@ class OfferData(unittest.TestCase): self.assertEqual(fd_to_send.tell(), 0) self.assertEqual(fd_to_send.read(), message) - def test_broken_symlink(self): + def _create_broken_symlink(self): if not hasattr(os, 'symlink'): raise unittest.SkipTest("host OS does not support symlinks") @@ -81,9 +81,20 @@ class OfferData(unittest.TestCase): self.cfg.what = send_dir_arg self.cfg.cwd = parent_dir + def test_broken_symlink_raises_err(self): + self._create_broken_symlink() + self.cfg.ignore_unsendable_files = False e = self.assertRaises(UnsendableFileError, build_offer, self.cfg) self.assertEqual(str(e), "linky: No such file or directory") + def test_broken_symlink_is_ignored(self): + self._create_broken_symlink() + self.cfg.ignore_unsendable_files = True + d, fd_to_send = build_offer(self.cfg) + self.assertIn('(ignoring error)', self.cfg.stderr.getvalue()) + self.assertEqual(d['directory']['numfiles'], 0) + self.assertEqual(d['directory']['numbytes'], 0) + def test_missing_file(self): self.cfg.what = filename = "missing" send_dir = self.mktemp() From 8c2e739dcd9f9b3408efba21bf3850e63a08752b Mon Sep 17 00:00:00 2001 From: Atul Varma Date: Wed, 24 May 2017 11:52:56 -0400 Subject: [PATCH 4/4] Make test_broken_symlink_raises_err resilient across OS's/locales. --- src/wormhole/test/test_cli.py | 7 ++++++- 1 file changed, 6 insertions(+), 1 deletion(-) diff --git a/src/wormhole/test/test_cli.py b/src/wormhole/test/test_cli.py index c0eca38..ba9cfc1 100644 --- a/src/wormhole/test/test_cli.py +++ b/src/wormhole/test/test_cli.py @@ -85,7 +85,12 @@ class OfferData(unittest.TestCase): self._create_broken_symlink() self.cfg.ignore_unsendable_files = False e = self.assertRaises(UnsendableFileError, build_offer, self.cfg) - self.assertEqual(str(e), "linky: No such file or directory") + + # On english distributions of Linux, this will be + # "linky: No such file or directory", but the error may be + # different on Windows and other locales and/or Unix variants, so + # we'll just assert the part we know about. + self.assertIn("linky: ", str(e)) def test_broken_symlink_is_ignored(self): self._create_broken_symlink()