blob: 81175d38355ce7b7332212e54e1e89e1cb446ecf [file] [log] [blame]
Chris Sosa90c78502012-10-05 17:07:42 -07001# Copyright (c) 2012 The Chromium OS Authors. All rights reserved.
2# Use of this source code is governed by a BSD-style license that can be
3# found in the LICENSE file.
4
David Purselle19f83b2015-07-01 12:57:07 -07005"""This implements the entry point for the `cros` CLI toolset.
Don Garrett25f309a2014-03-19 14:02:12 -07006
David Purselle19f83b2015-07-01 12:57:07 -07007This script is invoked by chromite/bin/cros, which sets up the
David Pursellffb90042015-03-23 09:21:41 -07008proper execution environment and calls this module's main() function.
Don Garrett25f309a2014-03-19 14:02:12 -07009
10In turn, this script looks for a subcommand based on how it was invoked. For
David Pursellffb90042015-03-23 09:21:41 -070011example, `cros lint` will use the cli/cros/cros_lint.py subcommand.
Don Garrett25f309a2014-03-19 14:02:12 -070012
David Pursellffb90042015-03-23 09:21:41 -070013See cli/ for actual command implementations.
Don Garrett25f309a2014-03-19 14:02:12 -070014"""
15
Mike Frysinger383367e2014-09-16 15:06:17 -040016from __future__ import print_function
17
Ryan Cui47f80e42013-04-01 19:01:54 -070018import sys
19
David Pursellf1c27c12015-03-18 09:51:38 -070020from chromite.cli import command
Chris Sosab445f792012-10-11 15:26:39 -070021from chromite.lib import commandline
Ralph Nathan91874ca2015-03-19 13:29:41 -070022from chromite.lib import cros_logging as logging
Ryan Cui47f80e42013-04-01 19:01:54 -070023from chromite.lib import stats
Chris Sosa90c78502012-10-05 17:07:42 -070024
25
Brian Harring984988f2012-10-10 22:53:30 -070026def GetOptions(my_commands):
David Pursellffb90042015-03-23 09:21:41 -070027 """Returns the argparse to use for commandline parsing."""
Ralph Nathan38372f42015-05-19 15:56:25 -070028 parser = commandline.ArgumentParser(caching=True, default_log_level='notice')
David Pursellf1c27c12015-03-18 09:51:38 -070029 if not command:
Chris Sosa90c78502012-10-05 17:07:42 -070030 return parser
31
David Pursellffb90042015-03-23 09:21:41 -070032 subparsers = parser.add_subparsers(title='Subcommands')
Mike Frysingerd6e2df02014-11-26 02:55:04 -050033 for cmd_name, class_def in sorted(my_commands.iteritems(),
34 key=lambda x: x[0]):
David James57d82a22012-12-04 11:24:58 -080035 epilog = getattr(class_def, 'EPILOG', None)
36 sub_parser = subparsers.add_parser(
Ryan Cui8b864512013-01-29 15:40:24 -080037 cmd_name, description=class_def.__doc__, epilog=epilog,
Ryo Hashimoto8bc997b2014-01-22 18:46:17 +090038 caching=class_def.use_caching_options,
David James57d82a22012-12-04 11:24:58 -080039 formatter_class=commandline.argparse.RawDescriptionHelpFormatter)
Chris Sosa90c78502012-10-05 17:07:42 -070040 class_def.AddParser(sub_parser)
41
42 return parser
43
44
Ryan Cui47f80e42013-04-01 19:01:54 -070045def _RunSubCommand(subcommand):
46 """Helper function for testing purposes."""
Ryan Cuide21f482013-08-06 11:50:59 -070047 return subcommand.Run()
Ryan Cui47f80e42013-04-01 19:01:54 -070048
49
Mike Frysinger9ad5fab2013-05-30 13:37:26 -040050def main(argv):
David Purselldfbfbc82015-03-05 10:59:16 -080051 try:
David Pursellf1c27c12015-03-18 09:51:38 -070052 parser = GetOptions(command.ListCommands())
David Purselldfbfbc82015-03-05 10:59:16 -080053 # Cros currently does nothing without a subcmd. Print help if no args are
54 # specified.
55 if not argv:
56 parser.print_help()
57 return 1
58
59 namespace = parser.parse_args(argv)
David Pursellffb90042015-03-23 09:21:41 -070060 subcommand = namespace.command_class(namespace)
David Purselldfbfbc82015-03-05 10:59:16 -080061 with stats.UploadContext() as queue:
62 if subcommand.upload_stats:
David Pursellffb90042015-03-23 09:21:41 -070063 cmd_base = subcommand.options.command_class.command_name
David Purselldfbfbc82015-03-05 10:59:16 -080064 cmd_stats = stats.Stats.SafeInit(cmd_line=sys.argv, cmd_base=cmd_base)
65 if cmd_stats:
66 queue.put([cmd_stats, stats.StatsUploader.URL,
67 subcommand.upload_stats_timeout])
68 # TODO: to make command completion faster, send an interrupt signal to the
69 # stats uploader task after the subcommand completes.
Mike Frysinger684d36e2015-06-03 05:03:34 -040070 try:
71 code = _RunSubCommand(subcommand)
72 except (commandline.ChrootRequiredError, commandline.ExecRequiredError):
73 # The higher levels want these passed back, so oblige.
74 raise
75 except Exception as e:
76 code = 1
77 logging.error('%s %s failed before completing.',
78 command.GetToolset(),
79 subcommand.command_name)
80 if namespace.debug:
81 raise
82 else:
83 logging.error(e)
84
David Purselldfbfbc82015-03-05 10:59:16 -080085 if code is not None:
86 return code
87
88 return 0
89 except KeyboardInterrupt:
90 logging.debug('Aborted due to keyboard interrupt.')
Chris Sosa90c78502012-10-05 17:07:42 -070091 return 1