2012-10-25 21:37:15 +00:00
|
|
|
#!/usr/bin/env python3
|
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
#
|
|
|
|
# 2010-2012 Nico Schottelius (nico-cdist at schottelius.org)
|
|
|
|
#
|
|
|
|
# This file is part of cdist.
|
|
|
|
#
|
|
|
|
# cdist is free software: you can redistribute it and/or modify
|
|
|
|
# it under the terms of the GNU General Public License as published by
|
|
|
|
# the Free Software Foundation, either version 3 of the License, or
|
|
|
|
# (at your option) any later version.
|
|
|
|
#
|
|
|
|
# cdist is distributed in the hope that it will be useful,
|
|
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
# GNU General Public License for more details.
|
|
|
|
#
|
|
|
|
# You should have received a copy of the GNU General Public License
|
|
|
|
# along with cdist. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
#
|
|
|
|
#
|
|
|
|
|
|
|
|
def commandline():
|
|
|
|
"""Parse command line"""
|
|
|
|
import argparse
|
|
|
|
|
|
|
|
import cdist.banner
|
|
|
|
import cdist.config
|
2013-08-07 07:24:10 +00:00
|
|
|
# import cdist.install
|
|
|
|
import cdist.shell
|
2012-10-25 21:37:15 +00:00
|
|
|
|
|
|
|
# Construct parser others can reuse
|
|
|
|
parser = {}
|
|
|
|
# Options _all_ parsers have in common
|
|
|
|
parser['loglevel'] = argparse.ArgumentParser(add_help=False)
|
|
|
|
parser['loglevel'].add_argument('-d', '--debug',
|
|
|
|
help='Set log level to debug', action='store_true',
|
|
|
|
default=False)
|
|
|
|
parser['loglevel'].add_argument('-v', '--verbose',
|
|
|
|
help='Set log level to info, be more verbose',
|
|
|
|
action='store_true', default=False)
|
|
|
|
|
|
|
|
# Main subcommand parser
|
|
|
|
parser['main'] = argparse.ArgumentParser(description='cdist ' + cdist.VERSION,
|
|
|
|
parents=[parser['loglevel']])
|
|
|
|
parser['main'].add_argument('-V', '--version',
|
|
|
|
help='Show version', action='version',
|
|
|
|
version='%(prog)s ' + cdist.VERSION)
|
|
|
|
parser['sub'] = parser['main'].add_subparsers(title="Commands")
|
|
|
|
|
|
|
|
# Banner
|
|
|
|
parser['banner'] = parser['sub'].add_parser('banner',
|
|
|
|
parents=[parser['loglevel']])
|
|
|
|
parser['banner'].set_defaults(func=cdist.banner.banner)
|
|
|
|
|
|
|
|
# Config and install (common stuff)
|
|
|
|
parser['configinstall'] = argparse.ArgumentParser(add_help=False)
|
|
|
|
parser['configinstall'].add_argument('host', nargs='+',
|
|
|
|
help='one or more hosts to operate on')
|
2012-10-30 14:59:02 +00:00
|
|
|
parser['configinstall'].add_argument('-c', '--conf-dir',
|
|
|
|
help='Add configuration directory (can be repeated, last one wins)',
|
|
|
|
action='append')
|
2012-10-25 21:37:15 +00:00
|
|
|
parser['configinstall'].add_argument('-i', '--initial-manifest',
|
|
|
|
help='Path to a cdist manifest or \'-\' to read from stdin.',
|
|
|
|
dest='manifest', required=False)
|
2013-05-02 14:41:16 +00:00
|
|
|
parser['configinstall'].add_argument('-n', '--dry-run',
|
|
|
|
help='Do not execute code', action='store_true')
|
2012-10-25 21:37:15 +00:00
|
|
|
parser['configinstall'].add_argument('-p', '--parallel',
|
|
|
|
help='Operate on multiple hosts in parallel',
|
|
|
|
action='store_true', dest='parallel')
|
|
|
|
parser['configinstall'].add_argument('-s', '--sequential',
|
|
|
|
help='Operate on multiple hosts sequentially (default)',
|
|
|
|
action='store_false', dest='parallel')
|
|
|
|
|
|
|
|
parser['configinstall'].add_argument('--remote-copy',
|
|
|
|
help='Command to use for remote copy (should behave like scp)',
|
|
|
|
action='store', dest='remote_copy',
|
2013-08-07 15:52:34 +00:00
|
|
|
default=cdist.REMOTE_COPY)
|
2012-10-25 21:37:15 +00:00
|
|
|
parser['configinstall'].add_argument('--remote-exec',
|
|
|
|
help='Command to use for remote execution (should behave like ssh)',
|
|
|
|
action='store', dest='remote_exec',
|
2013-08-07 15:52:34 +00:00
|
|
|
default=cdist.REMOTE_EXEC)
|
2012-10-25 21:37:15 +00:00
|
|
|
|
|
|
|
# Config
|
|
|
|
parser['config'] = parser['sub'].add_parser('config',
|
|
|
|
parents=[parser['loglevel'], parser['configinstall']])
|
2013-08-07 07:24:10 +00:00
|
|
|
parser['config'].set_defaults(func=cdist.config.Config.commandline)
|
|
|
|
|
|
|
|
# Shell
|
|
|
|
parser['shell'] = parser['sub'].add_parser('shell',
|
|
|
|
parents=[parser['loglevel']])
|
2013-08-07 15:52:34 +00:00
|
|
|
parser['shell'].add_argument('-s', '--shell',
|
|
|
|
help='Select shell to use, defaults to current shell')
|
2013-08-07 07:24:10 +00:00
|
|
|
parser['shell'].set_defaults(func=cdist.shell.Shell.commandline)
|
|
|
|
|
2012-10-25 21:37:15 +00:00
|
|
|
|
|
|
|
# Install
|
|
|
|
# 20120525/sar: commented until it actually does something
|
|
|
|
#parser['install'] = parser['sub'].add_parser('install',
|
|
|
|
# parents=[parser['loglevel'], parser['configinstall']])
|
|
|
|
#parser['install'].set_defaults(func=install)
|
|
|
|
|
|
|
|
for p in parser:
|
|
|
|
parser[p].epilog = "Get cdist at http://www.nico.schottelius.org/software/cdist/"
|
|
|
|
|
|
|
|
args = parser['main'].parse_args(sys.argv[1:])
|
|
|
|
|
|
|
|
# Loglevels are handled globally in here and debug wins over verbose
|
|
|
|
if args.verbose:
|
|
|
|
logging.root.setLevel(logging.INFO)
|
|
|
|
if args.debug:
|
|
|
|
logging.root.setLevel(logging.DEBUG)
|
|
|
|
|
|
|
|
log.debug(args)
|
Remove ugly argumentparser bug
Before:
[21:09] bento:~% cdist
Traceback (most recent call last):
File "/home/users/nico/p/cdist/cdist/bin/../scripts/cdist", line 232, in <module>
commandline()
File "/home/users/nico/p/cdist/cdist/bin/../scripts/cdist", line 106, in commandline
args.func(args)
AttributeError: 'Namespace' object has no attribute 'func'
After:
[21:11] bento:~% cdist
usage: cdist [-h] [-d] [-v] [-V] {banner,config} ...
cdist 2.1.1-48-gfd72c60
optional arguments:
-h, --help show this help message and exit
-d, --debug Set log level to debug
-v, --verbose Set log level to info, be more verbose
-V, --version Show version
Commands:
{banner,config}
Get cdist at http://www.nico.schottelius.org/software/cdist/
[21:11] bento:~%
Signed-off-by: Nico Schottelius <nico@bento.schottelius.org>
2013-06-07 19:10:57 +00:00
|
|
|
|
|
|
|
# Work around python 3.3 bug:
|
|
|
|
# http://bugs.python.org/issue16308
|
|
|
|
# http://bugs.python.org/issue9253
|
2013-08-07 16:49:47 +00:00
|
|
|
|
|
|
|
# FIXME: catching AttributeError also hides
|
|
|
|
# real problems.. try a different way
|
|
|
|
|
|
|
|
# FIXME: we always print main help, not
|
|
|
|
# the help of the actual parser being used!
|
Remove ugly argumentparser bug
Before:
[21:09] bento:~% cdist
Traceback (most recent call last):
File "/home/users/nico/p/cdist/cdist/bin/../scripts/cdist", line 232, in <module>
commandline()
File "/home/users/nico/p/cdist/cdist/bin/../scripts/cdist", line 106, in commandline
args.func(args)
AttributeError: 'Namespace' object has no attribute 'func'
After:
[21:11] bento:~% cdist
usage: cdist [-h] [-d] [-v] [-V] {banner,config} ...
cdist 2.1.1-48-gfd72c60
optional arguments:
-h, --help show this help message and exit
-d, --debug Set log level to debug
-v, --verbose Set log level to info, be more verbose
-V, --version Show version
Commands:
{banner,config}
Get cdist at http://www.nico.schottelius.org/software/cdist/
[21:11] bento:~%
Signed-off-by: Nico Schottelius <nico@bento.schottelius.org>
2013-06-07 19:10:57 +00:00
|
|
|
try:
|
2013-08-07 16:49:47 +00:00
|
|
|
getattr(args, "func")
|
Remove ugly argumentparser bug
Before:
[21:09] bento:~% cdist
Traceback (most recent call last):
File "/home/users/nico/p/cdist/cdist/bin/../scripts/cdist", line 232, in <module>
commandline()
File "/home/users/nico/p/cdist/cdist/bin/../scripts/cdist", line 106, in commandline
args.func(args)
AttributeError: 'Namespace' object has no attribute 'func'
After:
[21:11] bento:~% cdist
usage: cdist [-h] [-d] [-v] [-V] {banner,config} ...
cdist 2.1.1-48-gfd72c60
optional arguments:
-h, --help show this help message and exit
-d, --debug Set log level to debug
-v, --verbose Set log level to info, be more verbose
-V, --version Show version
Commands:
{banner,config}
Get cdist at http://www.nico.schottelius.org/software/cdist/
[21:11] bento:~%
Signed-off-by: Nico Schottelius <nico@bento.schottelius.org>
2013-06-07 19:10:57 +00:00
|
|
|
except AttributeError:
|
|
|
|
parser['main'].print_help()
|
2013-08-07 16:49:47 +00:00
|
|
|
sys.exit(0)
|
|
|
|
|
|
|
|
args.func(args)
|
2012-10-25 21:37:15 +00:00
|
|
|
|
2013-08-07 07:24:10 +00:00
|
|
|
#def install(args):
|
|
|
|
# configinstall(args, mode=cdist.install.Install)
|
2012-10-25 21:37:15 +00:00
|
|
|
|
|
|
|
def emulator():
|
|
|
|
"""Prepare and run emulator"""
|
|
|
|
import cdist.emulator
|
|
|
|
emulator = cdist.emulator.Emulator(sys.argv)
|
|
|
|
return emulator.run()
|
|
|
|
|
|
|
|
if __name__ == "__main__":
|
|
|
|
# Sys is needed for sys.exit()
|
|
|
|
import sys
|
|
|
|
|
|
|
|
cdistpythonversion = '3.2'
|
|
|
|
if sys.version < cdistpythonversion:
|
|
|
|
print('Cdist requires Python >= ' + cdistpythonversion +
|
|
|
|
' on the source host.', file=sys.stderr)
|
|
|
|
sys.exit(1)
|
|
|
|
|
|
|
|
|
|
|
|
exit_code = 0
|
|
|
|
|
|
|
|
try:
|
|
|
|
import logging
|
|
|
|
import os
|
|
|
|
import re
|
|
|
|
import cdist
|
|
|
|
|
|
|
|
log = logging.getLogger("cdist")
|
|
|
|
logging.basicConfig(format='%(levelname)s: %(message)s')
|
|
|
|
|
|
|
|
if re.match("__", os.path.basename(sys.argv[0])):
|
|
|
|
emulator()
|
|
|
|
else:
|
|
|
|
commandline()
|
|
|
|
|
|
|
|
except KeyboardInterrupt:
|
2013-08-12 09:42:00 +00:00
|
|
|
pass
|
|
|
|
|
|
|
|
# FIXME: We always get exit code = 130
|
|
|
|
# exit_code = 2
|
|
|
|
# does not make a difference
|
2012-10-25 21:37:15 +00:00
|
|
|
|
|
|
|
except cdist.Error as e:
|
|
|
|
log.error(e)
|
|
|
|
exit_code = 1
|
|
|
|
|
2013-08-12 09:42:00 +00:00
|
|
|
#sys.exit(20)
|
|
|
|
#print("ok2 %s" % exit_code)
|
2012-10-25 21:37:15 +00:00
|
|
|
sys.exit(exit_code)
|