# Kickstart parser for splitting kickstart into elements
#
# Copyright (C) 2017 Red Hat, Inc. All rights reserved.
#
# This program 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 2 of the License, or
# (at your option) any later version.
#
# This program 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 this program. If not, see .
#
__all__ = ["SplitKickstartParser", "VALID_SECTIONS_ANACONDA"]
from pykickstart.parser import KickstartParser
from pykickstart.sections import Section
from pyanaconda.modules.boss.kickstart_manager.element import KickstartElement,\
TrackedKickstartElements
VALID_SECTIONS_ANACONDA = [
"%pre", "%pre-install", "%post", "%onerror", "%traceback", "%packages", "%addon"
]
class StoreSection(Section):
"""Section for storing section content and header line references.
Similarly as NullSection defines a section that parser will recognize (ie
will not raise an error). The section will pass itself to an object for storing
sections if supplied.
"""
allLines = True
def __init__(self, *args, **kwargs):
"""Create a new StoreSection instance.
You must pass a sectionOpen parameter (including a leading '%') for the
section to make it valid but just ignored. If you want to store the
content, supply a store argument.
Required kwargs:
sectionOpen - section name, including '%' starting character
Optional kwargs:
store - an instance of an object for storing the section
(SplitKickstartParser) which must provide
add_section(StoreSection) method
attributes:
header_lineno - section header line in kickstart file
args - section header parsed by KickstartParser (shlex)
lines - list of section body lines
"""
super().__init__(*args, **kwargs)
self.sectionOpen = kwargs.get("sectionOpen")
self._store = kwargs.get("store", None)
self.header_lineno = 0
self.args = []
self.lines = []
def handleHeader(self, lineno, args):
self.header_lineno = lineno
self.args = args
def handleLine(self, line):
self.lines.append(line)
def finalize(self):
if self._store is not None:
self._store.add_section(self)
self.header_lineno = 0
self.args = []
self.lines = []
class SplitKickstartParser(KickstartParser):
"""Kickstart parser for storing kickstart elements.
Stores kickstart elements (commands, sections, addons) with their line
number and file name references to kickstart file.
Does not do any actual command or section parsing (ie command syntax
checking).
:raises KickstartParseError: on invalid section
:raises KickstartError: on missing %include unless instantiated with
missing_include_is_fatal=False
"""
# file name to be used in case of parsing string if not supplied
unknown_filename = ""
def __init__(self, handler, valid_sections=None, missing_include_is_fatal=True):
"""Initialize the parser.
:param valid_sections: list of valid section names (including '%')
:type valid_sections: list(str)
:param missing_include_is_fatal: raise KickstartError if included file
is not found
:type missing_include_is_fatal: bool
"""
self._valid_sections = valid_sections or []
# calls setupSections
super().__init__(handler, missingIncludeIsFatal=missing_include_is_fatal)
self._current_ks_filename = self.unknown_filename
self._result = TrackedKickstartElements()
@property
def valid_sections(self):
"""List of valid kickstart sections"""
return list(self._valid_sections)
@valid_sections.setter
def valid_sections(self, value):
self._valid_sections = value
def split(self, filename):
"""Split the kickstart file into elements.
:param filename: name of kickstart file
:type filename: str
:return: object containing kickstart elements with references to
kickstart files
:rtype: KickstartElements
"""
with open(filename, "r") as f:
kickstart = f.read()
return self.split_from_string(kickstart, filename=filename)
def split_from_string(self, kickstart, filename=None):
"""Split the kickstart given as string into elements.
:param kickstart: kickstart to be split
:type kickstart: str
:param filename: filename to be used as file reference in the result
:type filename: str
:return: object containing kickstart elements with references to
kickstart
:rtype: KickstartElements
"""
self._reset()
self._current_ks_filename = filename or self.unknown_filename
self.readKickstartFromString(kickstart)
return self._result
def add_section(self, section):
"""Adds a StoreSection to the result."""
element = KickstartElement(section.args, section.lines,
section.header_lineno, self._current_ks_filename)
self._result.append(element)
def _reset(self):
self._result = TrackedKickstartElements()
self.setupSections()
def _handleInclude(self, f):
"""Overrides parent to keep track of include file names."""
parent_file = self._current_ks_filename
self._current_ks_filename = f
super()._handleInclude(f)
self._current_ks_filename = parent_file
def handleCommand(self, lineno, args):
"""Overrides parent method to store the command."""
element = KickstartElement(args, [self._line], lineno, self._current_ks_filename)
self._result.append(element)
def setupSections(self):
"""Overrides parent method to store sections."""
self._sections = {}
for section in self._valid_sections:
self.registerSection(StoreSection(self.handler,
sectionOpen=section,
store=self))