Source code for launch.frontend.parser

# Copyright 2019 Open Source Robotics Foundation, Inc.
# Copyright 2020 Open Avatar Inc.
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
#     http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.

"""Module for Parser class and parsing methods."""

import itertools
import os.path
import sys
import traceback
from typing import List
from typing import Optional
from typing import Set
from typing import Text
from typing import TextIO
from typing import Tuple
from typing import Type
from typing import TYPE_CHECKING
from typing import Union
import warnings

from .entity import Entity
from .expose import instantiate_action
from .parse_substitution import parse_if_substitutions
from .parse_substitution import parse_substitution
from .parse_substitution import replace_escaped_characters
from ..action import Action
from ..invalid_launch_file_error import InvalidLaunchFileError
from ..substitution import Substitution
from ..utilities.type_utils import NormalizedValueType
from ..utilities.type_utils import StrSomeValueType
from ..utilities.typing_file_path import FilePath

if sys.version_info >= (3, 8):
    import importlib.metadata as importlib_metadata
else:
    import importlib_metadata

if TYPE_CHECKING:
    from ..launch_description import LaunchDescription


[docs] class InvalidFrontendLaunchFileError(InvalidLaunchFileError): """Exception raised when the given frontend launch file is not valid.""" pass
[docs] class Parser: """ Abstract class for parsing launch actions, substitutions and descriptions. Implementations of the parser class, should override the load method. They could also override the parse_substitution and/or get_file_extensions methods, or not. load_launch_extensions, parse_action, parse_description, get_available_extensions, may_parse, is_filename_valid, get_parsers_from_filename and get_file_extensions_from_parsers are not supposed to be overridden. """ extensions_loaded = False frontend_parsers = None
[docs] @classmethod def load_launch_extensions(cls): """Load launch extensions, in order to get all the exposed substitutions and actions.""" if cls.extensions_loaded is False: entry_points = importlib_metadata.entry_points() if hasattr(entry_points, 'select'): groups = entry_points.select(group='launch.frontend.launch_extension') else: groups = entry_points.get('launch.frontend.launch_extension', []) for entry_point in groups: try: entry_point.load() except Exception: warnings.warn(f'Failed to load the launch extension: {entry_point.name}\n' f'{traceback.format_exc()}') cls.extensions_loaded = True
[docs] @classmethod def load_parser_implementations(cls): """Load all the available frontend entities.""" if cls.frontend_parsers is None: parsers = {} entry_points = importlib_metadata.entry_points() if hasattr(entry_points, 'select'): groups = entry_points.select(group='launch.frontend.parser') else: groups = entry_points.get('launch.frontend.parser', []) for entry_point in groups: try: parsers[entry_point.name] = entry_point.load() except Exception: warnings.warn(f'Failed to load the parser extension: {entry_point.name}\n' f'{traceback.format_exc()}') cls.frontend_parsers = dict(sorted(parsers.items()))
[docs] def parse_action(self, entity: Entity) -> Action: """Parse an action, using its registered parsing method.""" self.load_launch_extensions() return instantiate_action(entity, self)
[docs] def parse_substitution(self, value: Text) -> List[Substitution]: """Parse a substitution.""" return parse_substitution(value)
[docs] def parse_if_substitutions( self, value: StrSomeValueType ) -> NormalizedValueType: """See :py:func:`launch.frontend.parser.parse_if_substitutions`.""" return parse_if_substitutions(value)
[docs] def escape_characters(self, value: Text) -> Text: """Escape characters in strings.""" return replace_escaped_characters(value)
[docs] def parse_description(self, entity: Entity) -> 'LaunchDescription': """Parse a launch description.""" # Avoid recursive import from ..launch_description import LaunchDescription # noqa: F811 if entity.type_name != 'launch': raise RuntimeError("Expected 'launch' as root tag") deprecated = entity.get_attr('deprecated', optional=True) actions = [self.parse_action(child) for child in entity.children] return LaunchDescription(actions, deprecated_reason=deprecated)
[docs] @classmethod def get_available_extensions(cls) -> List[Text]: """Return the registered extensions.""" cls.load_parser_implementations() assert cls.frontend_parsers is not None return cls.frontend_parsers.keys()
[docs] @classmethod def is_extension_valid( cls, extension: Text, ) -> bool: """Return an entity loaded with a markup file.""" warnings.warn( 'Parser.is_extension_valid is deprecated, use Parser.is_filename_valid instead') cls.load_parser_implementations() assert cls.frontend_parsers is not None return extension in cls.frontend_parsers
[docs] @classmethod def get_parser_from_extension( cls, extension: Text, ) -> Optional[Type['Parser']]: """Return an entity loaded with a markup file.""" warnings.warn( 'Parser.get_parser_from_extension is deprecated, ' 'use Parser.get_parsers_from_filename instead') cls.load_parser_implementations() assert cls.frontend_parsers is not None try: return cls.frontend_parsers[extension] except KeyError: raise RuntimeError('Not recognized frontend implementation')
[docs] @classmethod def may_parse( cls, filename: Text, ) -> bool: """Return `True` if the filename is valid for this parser.""" return any(filename.endswith('.' + ext) for ext in cls.get_file_extensions())
[docs] @classmethod def is_filename_valid( cls, filename: Text, ) -> bool: """Return `True` if the filename is valid for any parser.""" cls.load_parser_implementations() assert cls.frontend_parsers is not None return any( parser.may_parse(filename) for parser in cls.frontend_parsers.values() )
[docs] @classmethod def get_parsers_from_filename( cls, filename: Text, ) -> List[Type['Parser']]: """Return a list of parsers which entity loaded with a markup file.""" cls.load_parser_implementations() assert cls.frontend_parsers is not None return [ parser for parser in cls.frontend_parsers.values() if parser.may_parse(filename) ]
[docs] @classmethod def get_file_extensions_from_parsers(cls) -> Set[Type['Parser']]: """Return a set of file extensions known to the parser implementations.""" cls.load_parser_implementations() assert cls.frontend_parsers is not None return set(itertools.chain.from_iterable( parser_extension.get_file_extensions() for parser_extension in cls.frontend_parsers.values() ))
[docs] @classmethod def load( cls, file: Union[FilePath, TextIO], ) -> Tuple[Entity, 'Parser']: """ Parse an Entity from a markup language-based launch file. Parsers are exposed and provided by available frontend implementations. To choose the right parser, it'll first attempt to infer the launch description format based on the filename extension, if any. If format inference fails, it'll try all available parsers one after the other. """ # Imported here, to avoid recursive import. cls.load_parser_implementations() fileobj: TextIO if isinstance(file, (str, bytes, os.PathLike)): fileobj = open(file, 'r') didopen = True else: fileobj = file didopen = False try: filename = getattr(fileobj, 'name', '') implementations = cls.get_parsers_from_filename(filename) assert cls.frontend_parsers is not None implementations += [ parser for parser in cls.frontend_parsers.values() if parser not in implementations ] exceptions = [] for implementation in implementations: try: return implementation.load(fileobj) except Exception as ex: exceptions.append(ex) fileobj.seek(0) # file extension without leading '.' extension = os.path.splitext(filename)[1][1:] extension = '' if not cls.is_filename_valid(filename) else extension raise InvalidFrontendLaunchFileError(extension, likely_errors=exceptions) finally: if didopen: fileobj.close()
[docs] @classmethod def get_file_extensions(cls) -> Set[Text]: """Return the set of file extensions known to this parser.""" return set()