Source code for robot.running.arguments.argumentparser

#  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.utils import JYTHON, PY2, is_string, split_from_equals
from robot.variables import is_assign

from .argumentspec import ArgumentSpec

# Move PythonArgumentParser to this module when Python 2 support is dropped.
if PY2:
    from .py2argumentparser import PythonArgumentParser
else:
    from .py3argumentparser import PythonArgumentParser

if JYTHON:
    from java.lang import Class
    from java.util import List, Map


class _ArgumentParser(object):

    def __init__(self, type='Keyword'):
        self._type = type

    def parse(self, source, name=None):
        raise NotImplementedError


[docs]class JavaArgumentParser(_ArgumentParser):
[docs] def parse(self, signatures, name=None): if not signatures: return self._no_signatures_arg_spec(name) elif len(signatures) == 1: return self._single_signature_arg_spec(signatures[0], name) else: return self._multi_signature_arg_spec(signatures, name)
def _no_signatures_arg_spec(self, name): # Happens when a class has no public constructors return self._format_arg_spec(name) def _single_signature_arg_spec(self, signature, name): varargs, kwargs = self._get_varargs_and_kwargs_support(signature.args) positional = len(signature.args) - int(varargs) - int(kwargs) return self._format_arg_spec(name, positional, varargs=varargs, kwargs=kwargs) def _get_varargs_and_kwargs_support(self, args): if not args: return False, False if self._is_varargs_type(args[-1]): return True, False if not self._is_kwargs_type(args[-1]): return False, False if len(args) > 1 and self._is_varargs_type(args[-2]): return True, True return False, True def _is_varargs_type(self, arg): return arg is List or isinstance(arg, Class) and arg.isArray() def _is_kwargs_type(self, arg): return arg is Map def _multi_signature_arg_spec(self, signatures, name): mina = maxa = len(signatures[0].args) for sig in signatures[1:]: argc = len(sig.args) mina = min(argc, mina) maxa = max(argc, maxa) return self._format_arg_spec(name, maxa, maxa-mina) def _format_arg_spec(self, name, positional=0, defaults=0, varargs=False, kwargs=False): positional = ['arg%d' % (i+1) for i in range(positional)] if defaults: defaults = {name: '' for name in positional[-defaults:]} else: defaults = {} return ArgumentSpec(name, self._type, positional_only=positional, var_positional='varargs' if varargs else None, var_named='kwargs' if kwargs else None, defaults=defaults)
class _ArgumentSpecParser(_ArgumentParser): def parse(self, argspec, name=None): spec = ArgumentSpec(name, self._type) named_only = False for arg in argspec: arg = self._validate_arg(arg) if spec.var_named: self._raise_invalid_spec('Only last argument can be kwargs.') elif isinstance(arg, tuple): arg, default = arg arg = self._add_arg(spec, arg, named_only) spec.defaults[arg] = default elif self._is_kwargs(arg): spec.var_named = self._format_kwargs(arg) elif self._is_varargs(arg): if named_only: self._raise_invalid_spec('Cannot have multiple varargs.') if not self._is_kw_only_separator(arg): spec.var_positional = self._format_varargs(arg) named_only = True elif spec.defaults and not named_only: self._raise_invalid_spec('Non-default argument after default ' 'arguments.') else: self._add_arg(spec, arg, named_only) return spec def _validate_arg(self, arg): raise NotImplementedError def _raise_invalid_spec(self, error): raise DataError('Invalid argument specification: %s' % error) def _is_kwargs(self, arg): raise NotImplementedError def _format_kwargs(self, kwargs): raise NotImplementedError def _is_kw_only_separator(self, arg): raise NotImplementedError def _is_varargs(self, arg): raise NotImplementedError def _format_varargs(self, varargs): raise NotImplementedError def _format_arg(self, arg): return arg def _add_arg(self, spec, arg, named_only=False): arg = self._format_arg(arg) target = spec.positional_or_named if not named_only else spec.named_only target.append(arg) return arg
[docs]class DynamicArgumentParser(_ArgumentSpecParser): def _validate_arg(self, arg): if isinstance(arg, tuple): if self._is_invalid_tuple(arg): self._raise_invalid_spec('Invalid argument "%s".' % (arg,)) if len(arg) == 1: return arg[0] return arg if '=' in arg: return tuple(arg.split('=', 1)) return arg def _is_invalid_tuple(self, arg): return (len(arg) > 2 or not is_string(arg[0]) or (arg[0].startswith('*') and len(arg) > 1)) def _is_kwargs(self, arg): return arg.startswith('**') def _format_kwargs(self, kwargs): return kwargs[2:] def _is_varargs(self, arg): return arg.startswith('*') def _is_kw_only_separator(self, arg): return arg == '*' def _format_varargs(self, varargs): return varargs[1:]
[docs]class UserKeywordArgumentParser(_ArgumentSpecParser): def _validate_arg(self, arg): arg, default = split_from_equals(arg) if not (is_assign(arg) or arg == '@{}'): self._raise_invalid_spec("Invalid argument syntax '%s'." % arg) if default is not None: return arg, default return arg def _is_kwargs(self, arg): return arg[0] == '&' def _format_kwargs(self, kwargs): return kwargs[2:-1] def _is_varargs(self, arg): return arg[0] == '@' def _is_kw_only_separator(self, arg): return arg == '@{}' def _format_varargs(self, varargs): return varargs[2:-1] def _format_arg(self, arg): return arg[2:-1]