All Downloads are FREE. Search and download functionalities are using the official Maven repository.

robotframework-2.7.7.src.robot.running.namespace.py Maven / Gradle / Ivy

The newest version!
#  Copyright 2008-2012 Nokia Siemens Networks Oyj
#
#  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.

import os
import sys
import copy

from robot import utils
from robot.errors import DataError
from robot.variables import GLOBAL_VARIABLES, is_scalar_var
from robot.common import UserErrorHandler
from robot.output import LOGGER
from robot.parsing.settings import Library, Variables, Resource

from .userkeyword import UserLibrary
from .importer import Importer, ImportCache
from .runkwregister import RUN_KW_REGISTER
from .handlers import _XTimesHandler
from .context import EXECUTION_CONTEXTS


STDLIB_NAMES = set(('BuiltIn', 'Collections', 'Dialogs', 'Easter',
                    'OperatingSystem', 'Remote', 'Reserved',
                    'Screenshot', 'String', 'Telnet', 'XML'))
IMPORTER = Importer()


class Namespace:
    _default_libraries = ('BuiltIn', 'Reserved', 'Easter')
    _deprecated_libraries = {'BuiltIn': 'DeprecatedBuiltIn',
                             'OperatingSystem': 'DeprecatedOperatingSystem'}
    _library_import_by_path_endings = ('.py', '.java', '.class', '/', os.sep)

    def __init__(self, suite, parent_vars):
        if suite is not None:
            LOGGER.info("Initializing namespace for test suite '%s'" % suite.longname)
        self.variables = self._create_variables(suite, parent_vars)
        self.suite = suite
        self.test = None
        self.uk_handlers = []
        self.library_search_order = []
        self._testlibs = {}
        self._imported_resource_files = ImportCache()
        self._imported_variable_files = ImportCache()

    def handle_imports(self):
        self._import_default_libraries()
        if self.suite.source:
            self._handle_imports(self.suite.imports)

    def _create_variables(self, suite, parent_vars):
        variables = _VariableScopes(suite, parent_vars)
        variables['${SUITE_NAME}'] = suite.longname
        variables['${SUITE_SOURCE}'] = suite.source
        variables['${SUITE_DOCUMENTATION}'] = suite.doc
        variables['${SUITE_METADATA}'] = suite.metadata.copy()
        return variables

    def _import_default_libraries(self):
        for name in self._default_libraries:
            self.import_library(name)

    def _handle_imports(self, import_settings):
        for item in import_settings:
            try:
                if not item.name:
                    raise DataError('%s setting requires a name' % item.type)
                self._import(item)
            except DataError, err:
                item.report_invalid_syntax(unicode(err))

    def _import(self, import_setting):
        action = {'Library': self._import_library,
                  'Resource': self._import_resource,
                  'Variables': self._import_variables}[import_setting.type]
        action(import_setting, self.variables.current)

    def import_resource(self, name, overwrite=True):
        self._import_resource(Resource(None, name), overwrite=overwrite)

    def _import_resource(self, import_setting, variables=None, overwrite=False):
        path = self._resolve_name(import_setting, variables)
        if overwrite or path not in self._imported_resource_files:
            resource = IMPORTER.import_resource(path)
            self.variables.set_from_variable_table(resource.variable_table,
                                                   overwrite)
            self._imported_resource_files[path] \
                = UserLibrary(resource.keyword_table.keywords, resource.source)
            self._handle_imports(resource.setting_table.imports)
        else:
            LOGGER.info("Resource file '%s' already imported by suite '%s'"
                        % (path, self.suite.longname))

    def import_variables(self, name, args, overwrite=False, variables=None):
        self._import_variables(Variables(None, name, args), variables, overwrite)

    def _import_variables(self, import_setting, variables, overwrite=False):
        path = self._resolve_name(import_setting, variables)
        args = self._resolve_args(import_setting, variables)
        if overwrite or (path, args) not in self._imported_variable_files:
            self._imported_variable_files.add((path,args))
            self.variables.set_from_file(path, args, overwrite)
        else:
            msg = "Variable file '%s'" % path
            if args:
                msg += " with arguments %s" % utils.seq2str2(args)
            LOGGER.info("%s already imported by suite '%s'"
                        % (msg, self.suite.longname))

    def import_library(self, name, args=None, alias=None, variables=None):
        self._import_library(Library(None, name, args=args, alias=alias), variables)

    def _import_library(self, import_setting, variables):
        name = self._resolve_name(import_setting, variables)
        lib = IMPORTER.import_library(name, import_setting.args,
                                      import_setting.alias, variables)
        if lib.name in self._testlibs:
            LOGGER.info("Test library '%s' already imported by suite '%s'"
                        % (lib.name, self.suite.longname))
            return
        self._testlibs[lib.name] = lib
        lib.start_suite()
        if self.test:
            lib.start_test()
        self._import_deprecated_standard_libs(lib.name)

    def _resolve_name(self, import_setting, variables):
        name = import_setting.name
        if variables:
            try:
                name = variables.replace_string(name)
            except DataError, err:
                self._raise_replacing_vars_failed(import_setting, err)
        return self._get_path(name, import_setting.directory, import_setting.type)

    def _raise_replacing_vars_failed(self, import_setting, err):
        raise DataError("Replacing variables from setting '%s' failed: %s"
                        % (import_setting.type, unicode(err)))

    def _get_path(self, name, basedir, type):
        if type == 'Library' and not self._is_library_by_path(name):
            return name.replace(' ', '')
        try:
            return self._resolve_path(name.replace('/', os.sep), basedir)
        except DataError:
            self._raise_imported_does_not_exist(type, name)

    def _is_library_by_path(self, path):
        return path.lower().endswith(self._library_import_by_path_endings)

    def _resolve_path(self, path, basedir):
        for base in [basedir] + sys.path:
            if not (base and os.path.isdir(base)):
                continue
            if not isinstance(base, unicode):
                base = utils.decode_from_system(base)
            ret = os.path.abspath(os.path.join(base, path))
            if os.path.isfile(ret):
                return ret
            if os.path.isdir(ret) and os.path.isfile(os.path.join(ret, '__init__.py')):
                return ret
        raise DataError

    def _raise_imported_does_not_exist(self, type, path):
        type = {'Library': 'Test library',
                'Variables': 'Variable file',
                'Resource': 'Resource file'}[type]
        raise DataError("%s '%s' does not exist." % (type, path))

    def _resolve_args(self, import_setting, variables):
        if not variables:
            return import_setting.args
        try:
            return variables.replace_list(import_setting.args)
        except DataError, err:
            self._raise_replacing_vars_failed(import_setting, err)

    def _import_deprecated_standard_libs(self, name):
        if name in self._deprecated_libraries:
            self.import_library(self._deprecated_libraries[name])

    def start_test(self, test):
        self.variables.start_test(test)
        self.test = test
        for lib in self._testlibs.values():
            lib.start_test()
        self.variables['${TEST_NAME}'] = test.name
        self.variables['${TEST_DOCUMENTATION}'] = test.doc
        self.variables['@{TEST_TAGS}'] = test.tags[:]

    def end_test(self):
        self.test = None
        self.variables.end_test()
        self.uk_handlers = []
        for lib in self._testlibs.values():
            lib.end_test()

    def set_test_status_before_teardown(self, message, status):
        self.variables['${TEST_MESSAGE}'] = message
        self.variables['${TEST_STATUS}'] = status

    def end_suite(self):
        self.suite = None
        self.variables.end_suite()
        for lib in self._testlibs.values():
            lib.end_suite()

    def start_user_keyword(self, handler):
        self.variables.start_uk(handler)
        self.uk_handlers.append(handler)

    def end_user_keyword(self):
        self.variables.end_uk()
        self.uk_handlers.pop()

    def get_library_instance(self, libname):
        try:
            return self._testlibs[libname.replace(' ', '')].get_instance()
        except KeyError:
            raise DataError("No library with name '%s' found." % libname)

    def get_handler(self, name):
        try:
            handler = self._get_handler(name)
            if handler is None:
                raise DataError("No keyword with name '%s' found." % name)
        except DataError, err:
            handler = UserErrorHandler(name, unicode(err))
        self._replace_variables_from_user_handlers(handler)
        return handler

    def _replace_variables_from_user_handlers(self, handler):
        if hasattr(handler, 'replace_variables'):
            handler.replace_variables(self.variables)

    def _get_handler(self, name):
        handler = None
        if not name:
            raise DataError('Keyword name cannot be empty.')
        if not isinstance(name, basestring):
            raise DataError('Keyword name must be a string.')
        if '.' in name:
            handler = self._get_explicit_handler(name)
        if not handler:
            handler = self._get_implicit_handler(name)
        if not handler:
            handler = self._get_bdd_style_handler(name)
        if not handler:
            handler = self._get_x_times_handler(name)
        return handler

    def _get_x_times_handler(self, name):
        if not self._is_old_x_times_syntax(name):
            return None
        return _XTimesHandler(self._get_handler('Repeat Keyword'), name)

    def _is_old_x_times_syntax(self, name):
        if not name.lower().endswith('x'):
            return False
        times = name[:-1].strip()
        if is_scalar_var(times):
            return True
        try:
            int(times)
        except ValueError:
            return False
        else:
            return True

    def _get_bdd_style_handler(self, name):
        for prefix in ['given ', 'when ', 'then ', 'and ']:
            if name.lower().startswith(prefix):
                handler = self._get_handler(name[len(prefix):])
                if handler:
                    handler = copy.copy(handler)
                    handler.name = name
                return handler
        return None

    def _get_implicit_handler(self, name):
        for method in [self._get_handler_from_test_case_file_user_keywords,
                       self._get_handler_from_resource_file_user_keywords,
                       self._get_handler_from_library_keywords]:
            handler = method(name)
            if handler:
                return handler
        return None

    def _get_handler_from_test_case_file_user_keywords(self, name):
        if self.suite.user_keywords.has_handler(name):
            return self.suite.user_keywords.get_handler(name)

    def _get_handler_from_resource_file_user_keywords(self, name):
        found = [lib.get_handler(name) for lib
                 in self._imported_resource_files.values()
                 if lib.has_handler(name)]
        if not found:
            return None
        if len(found) > 1:
            found = self._get_handler_based_on_library_search_order(found)
        if len(found) == 1:
            return found[0]
        self._raise_multiple_keywords_found(name, found)

    def _get_handler_from_library_keywords(self, name):
        found = [lib.get_handler(name) for lib in self._testlibs.values()
                 if lib.has_handler(name)]
        if not found:
            return None
        if len(found) > 1:
            found = self._get_handler_based_on_library_search_order(found)
        if len(found) == 2:
            found = self._filter_stdlib_handler(found[0], found[1])
        if len(found) == 1:
            return found[0]
        self._raise_multiple_keywords_found(name, found)

    def _get_handler_based_on_library_search_order(self, handlers):
        for libname in self.library_search_order:
            for handler in handlers:
                if utils.eq(libname, handler.libname):
                    return [handler]
        return handlers

    def _filter_stdlib_handler(self, handler1, handler2):
        if handler1.library.orig_name in STDLIB_NAMES:
            standard, external = handler1, handler2
        elif handler2.library.orig_name in STDLIB_NAMES:
            standard, external = handler2, handler1
        else:
            return [handler1, handler2]
        if not RUN_KW_REGISTER.is_run_keyword(external.library.orig_name, external.name):
            LOGGER.warn(
                "Keyword '%s' found both from a user created test library "
                "'%s' and Robot Framework standard library '%s'. The user "
                "created keyword is used. To select explicitly, and to get "
                "rid of this warning, use either '%s' or '%s'."
                % (standard.name,
                   external.library.orig_name, standard.library.orig_name,
                   external.longname, standard.longname))
        return [external]

    def _get_explicit_handler(self, name):
        libname, kwname = name.rsplit('.', 1)
        # 1) Find matching lib(s)
        libs = [lib for lib
                in self._imported_resource_files.values() + self._testlibs.values()
                if utils.eq(lib.name, libname)]
        if not libs:
            return None
        # 2) Find matching kw from found libs
        found = [lib.get_handler(kwname) for lib in libs
                 if lib.has_handler(kwname)]
        if len(found) > 1:
            self._raise_multiple_keywords_found(name, found, implicit=False)
        return found and found[0] or None

    def _raise_multiple_keywords_found(self, name, found, implicit=True):
        error = "Multiple keywords with name '%s' found.\n" % name
        if implicit:
            error += "Give the full name of the keyword you want to use.\n"
        names = sorted(handler.longname for handler in found)
        error += "Found: %s" % utils.seq2str(names)
        raise DataError(error)


class _VariableScopes:

    def __init__(self, suite, parent_vars):
        # suite and parent are None only when used by copy_all
        if suite is not None:
            suite.variables.update(GLOBAL_VARIABLES)
            self._suite = self.current = suite.variables
        else:
            self._suite = self.current = None
        self._parents = []
        if parent_vars is not None:
            self._parents.append(parent_vars.current)
            self._parents.extend(parent_vars._parents)
        self._test = None
        self._uk_handlers = []

    def __len__(self):
        if self.current:
            return len(self.current)
        return 0

    def copy_all(self):
        vs = _VariableScopes(None, None)
        vs._suite = self._suite
        vs._test = self._test
        vs._uk_handlers = self._uk_handlers[:]
        vs._parents = self._parents[:]
        vs.current = self.current
        return vs

    def replace_list(self, items):
        return self.current.replace_list(items)

    def replace_scalar(self, items):
        return self.current.replace_scalar(items)

    def replace_string(self, string):
        return self.current.replace_string(string)

    def replace_run_kw_info(self, args, needed=sys.maxint):
        # @{list} variables can contain more or less arguments than needed.
        # Therefore we need to go through arguments one by one.
        processed = []
        while len(processed) < needed and args:
            processed.extend(self.current.replace_list([args.pop(0)]))
        # If @{list} variable is opened, arguments going further must be
        # escaped to prevent them being un-escaped twice.
        if len(processed) > needed:
            processed[needed:] = [self._escape_run_kw_arg(arg)
                                    for arg in processed[needed:]]
        return processed + args

    def _escape_run_kw_arg(self, arg):
        arg = utils.escape(arg)
        # Escape also special syntax used by Run Kw If and Run Kws.
        if arg in ('ELSE', 'ELSE IF', 'AND'):
            arg = '\\' + arg
        return arg

    def set_from_file(self, path, args, overwrite=False):
        variables = self._suite.set_from_file(path, args, overwrite)
        if self._test is not None:
            self._test._set_from_file(variables, overwrite=True, path=path)
        for varz in self._uk_handlers:
            varz._set_from_file(variables, overwrite=True, path=path)
        if self._uk_handlers:
            self.current._set_from_file(variables, overwrite=True, path=path)

    def set_from_variable_table(self, rawvariables, overwrite=False):
        self._suite.set_from_variable_table(rawvariables, overwrite)
        if self._test is not None:
            self._test.set_from_variable_table(rawvariables, overwrite)
        for varz in self._uk_handlers:
            varz.set_from_variable_table(rawvariables, overwrite)
        if self._uk_handlers:
            self.current.set_from_variable_table(rawvariables, overwrite)

    def replace_meta(self, name, item, errors):
        error = None
        for varz in [self.current] + self._parents:
            try:
                if name == 'Documentation':
                    return varz.replace_string(item, ignore_errors=True)
                elif isinstance(item, basestring):
                    return varz.replace_string(item)
                return varz.replace_list(item)
            except DataError, error:
                pass
        errors.append("Replacing variables from setting '%s' failed: %s"
                      % (name, error))
        return utils.unescape(item)

    def __getitem__(self, name):
        return self.current[name]

    def __setitem__(self, name, value):
        self.current[name] = value

    def end_suite(self):
        self._suite = self._test = self.current = None

    def start_test(self, test):
        self._test = self.current = self._suite.copy()

    def end_test(self):
        self.current = self._suite

    def start_uk(self, handler):
        self._uk_handlers.append(self.current)
        self.current = self.current.copy()

    def end_uk(self):
        self.current = self._uk_handlers.pop()

    def set_global(self, name, value):
        GLOBAL_VARIABLES.__setitem__(name, value)
        for ns in EXECUTION_CONTEXTS.namespaces:
            ns.variables.set_suite(name, value)

    def set_suite(self, name, value):
        self._suite.__setitem__(name, value)
        self.set_test(name, value, False)

    def set_test(self, name, value, fail_if_no_test=True):
        if self._test is not None:
            self._test.__setitem__(name, value)
        elif fail_if_no_test:
            raise DataError("Cannot set test variable when no test is started")
        for varz in self._uk_handlers:
            varz.__setitem__(name, value)
        self.current.__setitem__(name, value)

    def keys(self):
        return self.current.keys()

    def has_key(self, key):
        return self.current.has_key(key)

    __contains__ = has_key

    def contains(self, name, extended=False):
        return self.current.contains(name, extended)




© 2015 - 2024 Weber Informatics LLC | Privacy Policy