fix logfile specified a directory
[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, re
19 from optparse import SUPPRESS_HELP
20
21 from mic import msger, rt_util
22 from mic.utils import cmdln, errors, rpmmisc
23 from conf import configmgr
24 from plugin import pluginmgr
25
26 class Creator(cmdln.Cmdln):
27     """${name}: create an image
28
29     Usage:
30         ${name} SUBCOMMAND <ksfile> [OPTS]
31
32     ${command_list}
33     ${option_list}
34     """
35
36     name = 'mic create(cr)'
37
38     def __init__(self, *args, **kwargs):
39         cmdln.Cmdln.__init__(self, *args, **kwargs)
40         self._subcmds = []
41
42         # get cmds from pluginmgr
43         # mix-in do_subcmd interface
44         for subcmd, klass in pluginmgr.get_plugins('imager').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             self._subcmds.append(subcmd)
52
53     def get_optparser(self):
54         optparser = cmdln.CmdlnOptionParser(self)
55         optparser.add_option('-d', '--debug', action='store_true',
56                              dest='debug',
57                              help=SUPPRESS_HELP)
58         optparser.add_option('-v', '--verbose', action='store_true',
59                              dest='verbose',
60                              help=SUPPRESS_HELP)
61         optparser.add_option('', '--logfile', type='string', dest='logfile',
62                              default=None,
63                              help='Path of logfile')
64         optparser.add_option('-c', '--config', type='string', dest='config',
65                              default=None,
66                              help='Specify config file for mic')
67         optparser.add_option('-k', '--cachedir', type='string', action='store',
68                              dest='cachedir', default=None,
69                              help='Cache directory to store the downloaded')
70         optparser.add_option('-o', '--outdir', type='string', action='store',
71                              dest='outdir', default=None,
72                              help='Output directory')
73         optparser.add_option('-A', '--arch', type='string', dest='arch',
74                              default=None,
75                              help='Specify repo architecture')
76         optparser.add_option('', '--release', type='string', dest='release',
77                              default=None, metavar='RID',
78                              help='Generate a release of RID with all necessary'
79                                   ' files, when @BUILD_ID@ is contained in '
80                                   'kickstart file, it will be replaced by RID')
81         optparser.add_option("", "--record-pkgs", type="string",
82                              dest="record_pkgs", default=None,
83                              help='Record the info of installed packages, '
84                                   'multiple values can be specified which '
85                                   'joined by ",", valid values: "name", '
86                                   '"content", "license"')
87         optparser.add_option('', '--pkgmgr', type='string', dest='pkgmgr',
88                              default=None,
89                              help='Specify backend package manager')
90         optparser.add_option('', '--local-pkgs-path', type='string',
91                              dest='local_pkgs_path', default=None,
92                              help='Path for local pkgs(rpms) to be installed')
93         optparser.add_option('', '--runtime', type='string',
94                              dest='runtime', default=None,
95                              help='Specify  runtime mode, avaiable: bootstrap, native')
96         # --taring-to is alias to --pack-to
97         optparser.add_option('', '--taring-to', type='string',
98                              dest='pack_to', default=None,
99                              help=SUPPRESS_HELP)
100         optparser.add_option('', '--pack-to', type='string',
101                              dest='pack_to', default=None,
102                              help='Pack the images together into the specified'
103                                   ' achive, extension supported: .zip, .tar, '
104                                   '.tar.gz, .tar.bz2, etc. by default, .tar '
105                                   'will be used')
106         optparser.add_option('', '--copy-kernel', action='store_true',
107                              dest='copy_kernel',
108                              help='Copy kernel files from image /boot directory'
109                                   ' to the image output directory.')
110         optparser.add_option('', '--repourl', action='append',
111                              dest='repourl', default=[],
112                              help=SUPPRESS_HELP)
113         return optparser
114
115     def preoptparse(self, argv):
116         optparser = self.get_optparser()
117
118         largs = []
119         rargs = []
120         while argv:
121             arg = argv.pop(0)
122
123             if arg in ('-h', '--help'):
124                 rargs.append(arg)
125
126             elif optparser.has_option(arg):
127                 largs.append(arg)
128
129                 if optparser.get_option(arg).takes_value():
130                     try:
131                         largs.append(argv.pop(0))
132                     except IndexError:
133                         raise errors.Usage("option %s requires arguments" % arg)
134
135             else:
136                 if arg.startswith("--"):
137                     if "=" in arg:
138                         opt = arg.split("=")[0]
139                     else:
140                         opt = None
141                 elif arg.startswith("-") and len(arg) > 2:
142                     opt = arg[0:2]
143                 else:
144                     opt = None
145
146                 if opt and optparser.has_option(opt):
147                     largs.append(arg)
148                 else:
149                     rargs.append(arg)
150
151         return largs + rargs
152
153     def postoptparse(self):
154         abspath = lambda pth: os.path.abspath(os.path.expanduser(pth))
155
156         if self.options.verbose:
157             msger.set_loglevel('verbose')
158         if self.options.debug:
159             msger.set_loglevel('debug')
160
161         if self.options.logfile:
162             logfile_abs_path = abspath(self.options.logfile)
163             if not os.path.exists(os.path.dirname(logfile_abs_path)):
164                 os.makedirs(os.path.dirname(logfile_abs_path))
165             if os.path.isdir(logfile_abs_path):
166                 raise errors.Usage("logfile's path %s should be file"
167                                    % self.options.logfile)
168             msger.set_interactive(False)
169             msger.set_logfile(logfile_abs_path)
170             configmgr.create['logfile'] = self.options.logfile
171
172         if self.options.config:
173             configmgr.reset()
174             configmgr._siteconf = self.options.config
175
176         if self.options.outdir is not None:
177             configmgr.create['outdir'] = abspath(self.options.outdir)
178         if self.options.cachedir is not None:
179             configmgr.create['cachedir'] = abspath(self.options.cachedir)
180         os.environ['ZYPP_LOCKFILE_ROOT'] = configmgr.create['cachedir']
181
182         if self.options.local_pkgs_path is not None:
183             if not os.path.exists(self.options.local_pkgs_path):
184                 msger.error('Local pkgs directory: \'%s\' not exist' \
185                               % self.options.local_pkgs_path)
186             configmgr.create['local_pkgs_path'] = self.options.local_pkgs_path
187
188         if self.options.release:
189             configmgr.create['release'] = self.options.release
190
191         if self.options.record_pkgs:
192             configmgr.create['record_pkgs'] = []
193             for infotype in self.options.record_pkgs.split(','):
194                 if infotype not in ('name', 'content', 'license'):
195                     raise errors.Usage('Invalid pkg recording: %s, valid ones:'
196                                        ' "name", "content", "license"' \
197                                        % infotype)
198
199                 configmgr.create['record_pkgs'].append(infotype)
200
201         if self.options.arch is not None:
202             supported_arch = sorted(rpmmisc.archPolicies.keys(), reverse=True)
203             if self.options.arch in supported_arch:
204                 configmgr.create['arch'] = self.options.arch
205             else:
206                 raise errors.Usage('Invalid architecture: "%s".\n'
207                                    '  Supported architectures are: \n'
208                                    '  %s' % (self.options.arch,
209                                                ', '.join(supported_arch)))
210
211         if self.options.pkgmgr is not None:
212             configmgr.create['pkgmgr'] = self.options.pkgmgr
213
214         if self.options.runtime:
215             configmgr.set_runtime(self.options.runtime)
216
217         if self.options.pack_to is not None:
218             configmgr.create['pack_to'] = self.options.pack_to
219
220         if self.options.copy_kernel:
221             configmgr.create['copy_kernel'] = self.options.copy_kernel
222
223         if self.options.repourl:
224             for item in self.options.repourl:
225                 try:
226                     key, val = item.split('=')
227                 except:
228                     continue
229                 configmgr.create['repourl'][key] = val
230
231     def main(self, argv=None):
232         if argv is None:
233             argv = sys.argv
234         else:
235             argv = argv[:] # don't modify caller's list
236
237         self.optparser = self.get_optparser()
238         if self.optparser:
239             try:
240                 argv = self.preoptparse(argv)
241                 self.options, args = self.optparser.parse_args(argv)
242
243             except cmdln.CmdlnUserError, ex:
244                 msg = "%s: %s\nTry '%s help' for info.\n"\
245                       % (self.name, ex, self.name)
246                 msger.error(msg)
247
248             except cmdln.StopOptionProcessing, ex:
249                 return 0
250         else:
251             # optparser=None means no process for opts
252             self.options, args = None, argv[1:]
253
254         if not args:
255             return self.emptyline()
256
257         self.postoptparse()
258
259         return self.cmd(args)
260
261     def precmd(self, argv): # check help before cmd
262         if len(argv) == 1:
263             return ['help', argv[0]]
264
265         if '-h' in argv or '?' in argv or '--help' in argv or 'help' in argv:
266             return argv
267
268         if os.geteuid() != 0:
269             raise msger.error("Root permission is required, abort")
270
271         return argv
272
273     def do_auto(self, subcmd, opts, *args):
274         """${cmd_name}: auto detect image type from magic header
275
276         Usage:
277             ${name} ${cmd_name} <ksfile>
278
279         ${cmd_option_list}
280         """
281         def parse_magic_line(re_str, pstr, ptype='mic'):
282             ptn = re.compile(re_str)
283             m = ptn.match(pstr)
284             if not m or not m.groups():
285                 return None
286
287             inline_argv = m.group(1).strip()
288             if ptype == 'mic':
289                 m2 = re.search('(?P<format>\w+)', inline_argv)
290             elif ptype == 'mic2':
291                 m2 = re.search('(-f|--format(=)?)\s*(?P<format>\w+)',
292                                inline_argv)
293             else:
294                 return None
295
296             if m2:
297                 cmdname = m2.group('format')
298                 inline_argv = inline_argv.replace(m2.group(0), '')
299                 return (cmdname, inline_argv)
300
301             return None
302
303         if len(args) != 1:
304             raise errors.Usage("Extra arguments given")
305
306         if not os.path.exists(args[0]):
307             raise errors.CreatorError("Can't find the file: %s" % args[0])
308
309         with open(args[0], 'r') as rf:
310             first_line = rf.readline()
311
312         mic_re = '^#\s*-\*-mic-options-\*-\s+(.*)\s+-\*-mic-options-\*-'
313         mic2_re = '^#\s*-\*-mic2-options-\*-\s+(.*)\s+-\*-mic2-options-\*-'
314
315         result = parse_magic_line(mic_re, first_line, 'mic') \
316                  or parse_magic_line(mic2_re, first_line, 'mic2')
317         if not result:
318             raise errors.KsError("Invalid magic line in file: %s" % args[0])
319
320         if result[0] not in self._subcmds:
321             raise errors.KsError("Unsupport format '%s' in %s"
322                                  % (result[0], args[0]))
323
324         argv = ' '.join(result + args).split()
325         self.main(argv)
326