make the configmgr better
[tools/mic.git] / mic / creator.py
1 #!/usr/bin/python -tt
2 #
3 # Copyright (c) 2011 Intel, Inc.
4 #
5 # This program is free software; you can redistribute it and/or modify it
6 # under the terms of the GNU General Public License as published by the Free
7 # Software Foundation; version 2 of the License
8 #
9 # This program is distributed in the hope that it will be useful, but
10 # WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
11 # or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
12 # for more details.
13 #
14 # You should have received a copy of the GNU General Public License along
15 # with this program; if not, write to the Free Software Foundation, Inc., 59
16 # Temple Place - Suite 330, Boston, MA 02111-1307, USA.
17
18 import os, sys
19 from optparse import SUPPRESS_HELP
20
21 from mic import pluginmgr, msger
22 from mic.utils import cmdln, errors, rpmmisc
23 from conf import configmgr
24
25 class Creator(cmdln.Cmdln):
26     """${name}: create an image
27
28     Usage:
29         ${name} SUBCOMMAND [OPTS] [ARGS..]
30
31     ${command_list}
32     ${option_list}
33     """
34
35     name = 'mic create(cr)'
36
37     def __init__(self, *args, **kwargs):
38         cmdln.Cmdln.__init__(self, *args, **kwargs)
39
40         # get cmds from pluginmgr
41         self.plugincmds = pluginmgr.PluginMgr().get_plugins('imager')
42
43         # mix-in do_subcmd interface
44         for subcmd, klass in self.plugincmds.iteritems():
45             if not hasattr(klass, 'do_create'):
46                 msger.warning("Unsurpport subcmd: %s" % subcmd)
47                 continue
48
49             func = getattr(klass, 'do_create')
50             setattr(self.__class__, "do_"+subcmd, func)
51
52     def get_optparser(self):
53         optparser = cmdln.CmdlnOptionParser(self)
54         optparser.add_option('-d', '--debug', action='store_true', dest='debug', help=SUPPRESS_HELP)
55         optparser.add_option('-v', '--verbose', action='store_true', dest='verbose', help=SUPPRESS_HELP)
56         optparser.add_option('', '--logfile', type='string', dest='logfile', default=None, help='Path of logfile')
57         optparser.add_option('-c', '--config', type='string', dest='config', default=None, help='Specify config file for mic')
58         optparser.add_option('-k', '--cachedir', type='string', action='store', dest='cachedir', default=None, help='Cache directory to store the downloaded')
59         optparser.add_option('-o', '--outdir', type='string', action='store', dest='outdir', default=None, help='Output directory')
60         optparser.add_option('-A', '--arch', type='string', dest='arch', default=None, help='Specify repo architecture')
61         optparser.add_option('', '--release', type='string', dest='release', default=None, metavar='RID', help='Generate a release of RID with all necessary files, when @BUILD_ID@ is contained in kickstart file, it will be replaced by RID')
62         optparser.add_option("", "--record-pkgs", type="string", dest="record_pkgs", default=None,
63                              help='Record the info of installed packages, multiple values can be specified which joined by ",", valid values: "name", "content", "license"')
64         optparser.add_option('', '--pkgmgr', type='string', dest='pkgmgr', default=None, help='Specify backend package manager')
65         optparser.add_option('', '--local-pkgs-path', type='string', dest='local_pkgs_path', default=None, help='Path for local pkgs(rpms) to be installed')
66         return optparser
67
68     def preoptparse(self, argv):
69         optparser = self.get_optparser()
70
71         largs = []
72         rargs = []
73         while argv:
74             arg = argv.pop(0)
75
76             if arg in ('-h', '--help'):
77                 rargs.append(arg)
78
79             elif optparser.has_option(arg):
80                 largs.append(arg)
81
82                 if optparser.get_option(arg).takes_value():
83                     try:
84                         largs.append(argv.pop(0))
85                     except IndexError:
86                         raise errors.Usage("%s option requires an argument" % arg)
87
88             else:
89                 if arg.startswith("--"):
90                     if "=" in arg:
91                         opt = arg.split("=")[0]
92                     else:
93                         opt = None
94                 elif arg.startswith("-") and len(arg) > 2:
95                     opt = arg[0:2]
96                 else:
97                     opt = None
98
99                 if opt and optparser.has_option(opt):
100                     largs.append(arg)
101                 else:
102                     rargs.append(arg)
103
104         return largs + rargs
105
106     def postoptparse(self):
107         if self.options.verbose:
108             msger.set_loglevel('verbose')
109         if self.options.debug:
110             msger.set_loglevel('debug')
111
112         if self.options.logfile:
113             msger.set_interactive(False)
114             msger.set_logfile(self.options.logfile)
115             configmgr.create['logfile'] = self.options.logfile
116
117         if self.options.config:
118             configmgr.reset()
119             configmgr._siteconf = self.options.config
120
121         if self.options.outdir is not None:
122             configmgr.create['outdir'] = self.options.outdir
123         if self.options.cachedir is not None:
124             configmgr.create['cachedir'] = self.options.cachedir
125         if self.options.local_pkgs_path is not None:
126             configmgr.create['local_pkgs_path'] = self.options.local_pkgs_path
127
128         if self.options.release:
129             configmgr.create['release'] = self.options.release
130
131         if self.options.record_pkgs:
132             configmgr.create['record_pkgs'] = []
133             for infotype in self.options.record_pkgs.split(','):
134                 if infotype not in ('name', 'content', 'license'):
135                     raise errors.Usage('Invalid pkg recording: %s, valid ones: "name", "content", "license"' % infotype)
136
137                 configmgr.create['record_pkgs'].append(infotype)
138
139         if self.options.arch is not None:
140             supported_arch = sorted(rpmmisc.archPolicies.keys(), reverse=True)
141             if self.options.arch in supported_arch:
142                 configmgr.create['arch'] = self.options.arch
143             else:
144                 raise errors.Usage('Invalid architecture: "%s".\n' \
145                                    '  Supported architectures are: \n' \
146                                    '  %s\n' % (self.options.arch, ', '.join(supported_arch)))
147
148         if self.options.pkgmgr is not None:
149             configmgr.create['pkgmgr'] = self.options.pkgmgr
150
151     def main(self, argv=None):
152         if argv is None:
153             argv = sys.argv
154         else:
155             argv = argv[:] # don't modify caller's list
156
157         self.optparser = self.get_optparser()
158         if self.optparser:
159             try:
160                 argv = self.preoptparse(argv)
161                 self.options, args = self.optparser.parse_args(argv)
162
163             except cmdln.CmdlnUserError, ex:
164                 msg = "%s: %s\nTry '%s help' for info.\n"\
165                       % (self.name, ex, self.name)
166                 msger.error(msg)
167
168             except cmdln.StopOptionProcessing, ex:
169                 return 0
170         else:
171             # optparser=None means no process for opts
172             self.options, args = None, argv[1:]
173
174         self.postoptparse()
175
176         if not args:
177             return self.emptyline()
178
179         if os.geteuid() != 0:
180             msger.error('Root permission is required to continue, abort')
181
182         return self.cmd(args)
183