Source code for robot.result.flattenkeywordmatcher

#  Copyright 2008-2015 Nokia Networks
#  Copyright 2016-     Robot Framework Foundation
#
#  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.

from robot.errors import DataError
from robot.model import SuiteVisitor, TagPatterns
from robot.utils import html_escape, MultiMatcher

from .model import Keyword


[docs] def validate_flatten_keyword(options): for opt in options: low = opt.lower() # TODO: Deprecate 'foritem' in RF 7.4! if low == "foritem": low = "iteration" if not ( low in ("for", "while", "iteration") or low.startswith(("name:", "tag:")) ): raise DataError( f"Expected 'FOR', 'WHILE', 'ITERATION', 'TAG:<pattern>' or " f"'NAME:<pattern>', got '{opt}'." )
[docs] def create_flatten_message(original): if not original: start = "" elif original.startswith("*HTML*"): start = original[6:].strip() + "<hr>" else: start = html_escape(original) + "<hr>" return f'*HTML* {start}<span class="robot-note">Content flattened.</span>'
[docs] class FlattenByTypeMatcher: def __init__(self, flatten): if isinstance(flatten, str): flatten = [flatten] flatten = [f.lower() for f in flatten] self.types = set() if "for" in flatten: self.types.add("for") if "while" in flatten: self.types.add("while") if "iteration" in flatten or "foritem" in flatten: self.types.add("iter") # Matches output.xml tag. self.types.add("iteration") # Matches model object type.
[docs] def match(self, tag): return tag in self.types
def __bool__(self): return bool(self.types)
[docs] class FlattenByNameMatcher: def __init__(self, flatten): if isinstance(flatten, str): flatten = [flatten] names = [n[5:] for n in flatten if n[:5].lower() == "name:"] self._matcher = MultiMatcher(names)
[docs] def match(self, name, owner=None): name = f"{owner}.{name}" if owner else name return self._matcher.match(name)
def __bool__(self): return bool(self._matcher)
[docs] class FlattenByTagMatcher: def __init__(self, flatten): if isinstance(flatten, str): flatten = [flatten] patterns = [p[4:] for p in flatten if p[:4].lower() == "tag:"] self._matcher = TagPatterns(patterns)
[docs] def match(self, tags): return self._matcher.match(tags)
def __bool__(self): return bool(self._matcher)
[docs] class Flattener(SuiteVisitor): def __init__(self, flatten): self.name_matcher = FlattenByNameMatcher(flatten) self.tag_matcher = FlattenByTagMatcher(flatten) self.type_matcher = FlattenByTypeMatcher(flatten)
[docs] def start_suite(self, suite): return bool(self)
[docs] def start_keyword(self, kw): if self.name_matcher and self.name_matcher.match(kw.name, kw.owner): self._flatten(kw) if self.tag_matcher and self.tag_matcher.match(kw.tags): self._flatten(kw)
[docs] def start_body_item(self, item): if self.type_matcher and self.type_matcher.match(item.type.lower()): self._flatten(item)
def _flatten(self, item): item.message = create_flatten_message(item.message) item.body = MessageFinder(item).messages def __bool__(self): return bool(self.name_matcher or self.tag_matcher or self.type_matcher)
[docs] class FlattenByTags(SuiteVisitor): def __init__(self, flatten): if isinstance(flatten, str): flatten = [flatten] patterns = [p[4:] for p in flatten if p[:4].lower() == "tag:"] self.matcher = TagPatterns(patterns)
[docs] def start_suite(self, suite): return bool(self.matcher)
[docs] def start_keyword(self, keyword: Keyword): if self.matcher.match(keyword.tags): keyword.message = create_flatten_message(keyword.message) keyword.body = MessageFinder(keyword).messages
[docs] class MessageFinder(SuiteVisitor): def __init__(self, keyword: Keyword): self.messages = [] keyword.visit(self)
[docs] def visit_message(self, message): self.messages.append(message)
# TODO: Refactor this module in RF 7.4. # - Currently code working with XML tags and model objects is somewhat messy. # Either separate it better or make it more generic. # - MessageFinder API is now that nice. # - The module doesn't anymore contain only matchers so it should be renamed.