from __future__ import print_function from __future__ import absolute_import from __future__ import division from __future__ import unicode_literals from future.utils import PY2 if PY2: from future.builtins import filter, map, zip, ascii, chr, hex, input, next, oct, open, pow, round, super, bytes, dict, list, object, range, str, max, min # noqa: F401 import os, sys from six.moves import StringIO import six try: from allmydata.scripts.types_ import SubCommands except ImportError: pass from twisted.python import usage from twisted.internet import defer, task, threads from allmydata.scripts.common import get_default_nodedir from allmydata.scripts import debug, create_node, cli, \ admin, tahoe_run, tahoe_invite from allmydata.util.encodingutil import quote_local_unicode_path, argv_to_unicode from allmydata.util.eliotutil import ( opt_eliot_destination, opt_help_eliot_destinations, eliot_logging_service, ) from .. import ( __full_version__, ) _default_nodedir = get_default_nodedir() NODEDIR_HELP = ("Specify which Tahoe node directory should be used. The " "directory should either contain a full Tahoe node, or a " "file named node.url that points to some other Tahoe node. " "It should also contain a file named '" + os.path.join('private', 'aliases') + "' which contains the mapping from alias name to root " "dirnode URI.") if _default_nodedir: NODEDIR_HELP += " [default for most commands: " + quote_local_unicode_path(_default_nodedir) + "]" # XXX all this 'dispatch' stuff needs to be unified + fixed up _control_node_dispatch = { "run": tahoe_run.run, } process_control_commands = [ ("run", None, tahoe_run.RunOptions, "run a node without daemonizing"), ] # type: SubCommands class Options(usage.Options): # unit tests can override these to point at StringIO instances stdin = sys.stdin stdout = sys.stdout stderr = sys.stderr subCommands = ( create_node.subCommands + admin.subCommands + process_control_commands + debug.subCommands + cli.subCommands + tahoe_invite.subCommands ) optFlags = [ ["quiet", "q", "Operate silently."], ["version", "V", "Display version numbers."], ["version-and-path", None, "Display version numbers and paths to their locations."], ] optParameters = [ ["node-directory", "d", None, NODEDIR_HELP], ["wormhole-server", None, u"ws://wormhole.tahoe-lafs.org:4000/v1", "The magic wormhole server to use.", six.text_type], ["wormhole-invite-appid", None, u"tahoe-lafs.org/invite", "The appid to use on the wormhole server.", six.text_type], ] def opt_version(self): print(__full_version__, file=self.stdout) self.no_command_needed = True opt_version_and_path = opt_version opt_eliot_destination = opt_eliot_destination opt_help_eliot_destinations = opt_help_eliot_destinations def __str__(self): return ("\nUsage: tahoe [global-options] [command-options]\n" + self.getUsage()) synopsis = "\nUsage: tahoe [global-options]" # used only for subcommands def getUsage(self, **kwargs): t = usage.Options.getUsage(self, **kwargs) t = t.replace("Options:", "\nGlobal options:", 1) return t + "\nPlease run 'tahoe --help' for more details on each command.\n" def postOptions(self): if not hasattr(self, 'subOptions'): if not hasattr(self, 'no_command_needed'): raise usage.UsageError("must specify a command") sys.exit(0) create_dispatch = {} for module in (create_node,): create_dispatch.update(module.dispatch) # type: ignore def parse_options(argv, config=None): if not config: config = Options() try: config.parseOptions(argv) except usage.error as e: if six.PY2: # Exceptions must stringify to bytes on Python 2. raise usage.error(*( arg.encode("utf-8") if isinstance(arg, unicode) else arg for arg in e.args )) raise return config def parse_or_exit_with_explanation_with_config(config, argv, stdout, stderr): """ Parse Tahoe-LAFS CLI arguments and return a configuration object if they are valid. If they are invalid, write an explanation to ``stdout`` and exit. :param allmydata.scripts.runner.Options config: An instance of the argument-parsing class to use. :param [unicode] argv: The argument list to parse, including the name of the program being run as ``argv[0]``. :param stdout: The file-like object to use as stdout. :param stderr: The file-like object to use as stderr. :raise SystemExit: If there is an argument-parsing problem. :return: ``config``, after using it to parse the argument list. """ try: parse_options(argv[1:], config=config) except usage.error as e: # `parse_options` may have the side-effect of initializing a # "sub-option" of the given configuration, even if it ultimately # raises an exception. For example, `tahoe run --invalid-option` will # set `config.subOptions` to an instance of # `allmydata.scripts.tahoe_run.RunOptions` and then raise a # `usage.error` because `RunOptions` does not recognize # `--invalid-option`. If `run` itself had a sub-options then the same # thing could happen but with another layer of nesting. We can # present the user with the most precise information about their usage # error possible by finding the most "sub" of the sub-options and then # showing that to the user along with the usage error. c = config while hasattr(c, 'subOptions'): c = c.subOptions print(str(c), file=stdout) # On Python 2 the string may turn into a unicode string, e.g. the error # may be unicode, in which case it will print funny. Once we're on # Python 3 we can just drop the ensure_str(). print(six.ensure_str("%s: %s\n" % (sys.argv[0], e)), file=stdout) sys.exit(1) return config def dispatch(config, stdin=sys.stdin, stdout=sys.stdout, stderr=sys.stderr): command = config.subCommand so = config.subOptions if config['quiet']: stdout = StringIO() so.stdout = stdout so.stderr = stderr so.stdin = stdin if command in create_dispatch: f = create_dispatch[command] elif command in _control_node_dispatch: f = _control_node_dispatch[command] elif command in debug.dispatch: f = debug.dispatch[command] elif command in admin.dispatch: f = admin.dispatch[command] elif command in cli.dispatch: # these are blocking, and must be run in a thread f0 = cli.dispatch[command] f = lambda so: threads.deferToThread(f0, so) elif command in tahoe_invite.dispatch: f = tahoe_invite.dispatch[command] else: raise usage.UsageError() d = defer.maybeDeferred(f, so) # the calling convention for CLI dispatch functions is that they either: # 1: succeed and return rc=0 # 2: print explanation to stderr and return rc!=0 # 3: raise an exception that should just be printed normally # 4: return a Deferred that does 1 or 2 or 3 def _raise_sys_exit(rc): sys.exit(rc) d.addCallback(_raise_sys_exit) return d def _maybe_enable_eliot_logging(options, reactor): if options.get("destinations"): service = eliot_logging_service(reactor, options["destinations"]) # There is no Twisted "Application" around to hang this on so start # and stop it ourselves. service.startService() reactor.addSystemEventTrigger("after", "shutdown", service.stopService) # Pass on the options so we can dispatch the subcommand. return options PYTHON_3_WARNING = ("Support for Python 3 is an incomplete work-in-progress." " Use at your own risk.") def run(configFactory=Options, argv=sys.argv, stdout=sys.stdout, stderr=sys.stderr): """ Run a Tahoe-LAFS node. :param configFactory: A zero-argument callable which creates the config object to use to parse the argument list. :param [str] argv: The argument list to use to configure the run. :param stdout: The file-like object to use for stdout. :param stderr: The file-like object to use for stderr. :raise SystemExit: Always raised after the run is complete. """ if six.PY3: print(PYTHON_3_WARNING, file=stderr) if sys.platform == "win32": from allmydata.windows.fixups import initialize initialize() # doesn't return: calls sys.exit(rc) task.react( lambda reactor: _run_with_reactor( reactor, configFactory(), argv, stdout, stderr, ), ) def _setup_coverage(reactor, argv): """ If coverage measurement was requested, start collecting coverage measurements and arrange to record those measurements when the process is done. Coverage measurement is considered requested if ``"--coverage"`` is in ``argv`` (and it will be removed from ``argv`` if it is found). There should be a ``.coveragerc`` file in the working directory if coverage measurement is requested. This is only necessary to support multi-process coverage measurement, typically when the test suite is running, and with the pytest-based *integration* test suite (at ``integration/`` in the root of the source tree) foremost in mind. The idea is that if you are running Tahoe-LAFS in a configuration where multiple processes are involved - for example, a test process and a client node process, if you only measure coverage from the test process then you will fail to observe most Tahoe-LAFS code that is being run. This function arranges to have any Tahoe-LAFS process (such as that client node process) collect and report coverage measurements as well. """ # can we put this _setup_coverage call after we hit # argument-parsing? # ensure_str() only necessary on Python 2. if six.ensure_str('--coverage') not in sys.argv: return argv.remove('--coverage') try: import coverage except ImportError: raise RuntimeError( "The 'coveage' package must be installed to use --coverage" ) # this doesn't change the shell's notion of the environment, but # it makes the test in process_startup() succeed, which is the # goal here. os.environ["COVERAGE_PROCESS_START"] = '.coveragerc' # maybe-start the global coverage, unless it already got started cov = coverage.process_startup() if cov is None: cov = coverage.process_startup.coverage def write_coverage_data(): """ Make sure that coverage has stopped; internally, it depends on ataxit handlers running which doesn't always happen (Twisted's shutdown hook also won't run if os._exit() is called, but it runs more-often than atexit handlers). """ cov.stop() cov.save() reactor.addSystemEventTrigger('after', 'shutdown', write_coverage_data) def _run_with_reactor(reactor, config, argv, stdout, stderr): """ Run a Tahoe-LAFS node using the given reactor. :param reactor: The reactor to use. This implementation largely ignores this and lets the rest of the implementation pick its own reactor. Oops. :param twisted.python.usage.Options config: The config object to use to parse the argument list. :param [str] argv: The argument list to parse, *excluding* the name of the program being run. :param stdout: See ``run``. :param stderr: See ``run``. :return: A ``Deferred`` that fires when the run is complete. """ _setup_coverage(reactor, argv) argv = list(map(argv_to_unicode, argv)) d = defer.maybeDeferred( parse_or_exit_with_explanation_with_config, config, argv, stdout, stderr, ) d.addCallback(_maybe_enable_eliot_logging, reactor) d.addCallback(dispatch, stdout=stdout, stderr=stderr) def _show_exception(f): # when task.react() notices a non-SystemExit exception, it does # log.err() with the failure and then exits with rc=1. We want this # to actually print the exception to stderr, like it would do if we # weren't using react(). if f.check(SystemExit): return f # dispatch function handled it f.printTraceback(file=stderr) sys.exit(1) d.addErrback(_show_exception) return d if __name__ == "__main__": run()