| Index: third_party/google-endpoints/packaging/markers.py
|
| diff --git a/third_party/google-endpoints/packaging/markers.py b/third_party/google-endpoints/packaging/markers.py
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..5fdf510ca650160d099ef6e7e945ffdc2324c771
|
| --- /dev/null
|
| +++ b/third_party/google-endpoints/packaging/markers.py
|
| @@ -0,0 +1,301 @@
|
| +# This file is dual licensed under the terms of the Apache License, Version
|
| +# 2.0, and the BSD License. See the LICENSE file in the root of this repository
|
| +# for complete details.
|
| +from __future__ import absolute_import, division, print_function
|
| +
|
| +import operator
|
| +import os
|
| +import platform
|
| +import sys
|
| +
|
| +from pyparsing import ParseException, ParseResults, stringStart, stringEnd
|
| +from pyparsing import ZeroOrMore, Group, Forward, QuotedString
|
| +from pyparsing import Literal as L # noqa
|
| +
|
| +from ._compat import string_types
|
| +from .specifiers import Specifier, InvalidSpecifier
|
| +
|
| +
|
| +__all__ = [
|
| + "InvalidMarker", "UndefinedComparison", "UndefinedEnvironmentName",
|
| + "Marker", "default_environment",
|
| +]
|
| +
|
| +
|
| +class InvalidMarker(ValueError):
|
| + """
|
| + An invalid marker was found, users should refer to PEP 508.
|
| + """
|
| +
|
| +
|
| +class UndefinedComparison(ValueError):
|
| + """
|
| + An invalid operation was attempted on a value that doesn't support it.
|
| + """
|
| +
|
| +
|
| +class UndefinedEnvironmentName(ValueError):
|
| + """
|
| + A name was attempted to be used that does not exist inside of the
|
| + environment.
|
| + """
|
| +
|
| +
|
| +class Node(object):
|
| +
|
| + def __init__(self, value):
|
| + self.value = value
|
| +
|
| + def __str__(self):
|
| + return str(self.value)
|
| +
|
| + def __repr__(self):
|
| + return "<{0}({1!r})>".format(self.__class__.__name__, str(self))
|
| +
|
| + def serialize(self):
|
| + raise NotImplementedError
|
| +
|
| +
|
| +class Variable(Node):
|
| +
|
| + def serialize(self):
|
| + return str(self)
|
| +
|
| +
|
| +class Value(Node):
|
| +
|
| + def serialize(self):
|
| + return '"{0}"'.format(self)
|
| +
|
| +
|
| +class Op(Node):
|
| +
|
| + def serialize(self):
|
| + return str(self)
|
| +
|
| +
|
| +VARIABLE = (
|
| + L("implementation_version") |
|
| + L("platform_python_implementation") |
|
| + L("implementation_name") |
|
| + L("python_full_version") |
|
| + L("platform_release") |
|
| + L("platform_version") |
|
| + L("platform_machine") |
|
| + L("platform_system") |
|
| + L("python_version") |
|
| + L("sys_platform") |
|
| + L("os_name") |
|
| + L("os.name") | # PEP-345
|
| + L("sys.platform") | # PEP-345
|
| + L("platform.version") | # PEP-345
|
| + L("platform.machine") | # PEP-345
|
| + L("platform.python_implementation") | # PEP-345
|
| + L("python_implementation") | # undocumented setuptools legacy
|
| + L("extra")
|
| +)
|
| +ALIASES = {
|
| + 'os.name': 'os_name',
|
| + 'sys.platform': 'sys_platform',
|
| + 'platform.version': 'platform_version',
|
| + 'platform.machine': 'platform_machine',
|
| + 'platform.python_implementation': 'platform_python_implementation',
|
| + 'python_implementation': 'platform_python_implementation'
|
| +}
|
| +VARIABLE.setParseAction(lambda s, l, t: Variable(ALIASES.get(t[0], t[0])))
|
| +
|
| +VERSION_CMP = (
|
| + L("===") |
|
| + L("==") |
|
| + L(">=") |
|
| + L("<=") |
|
| + L("!=") |
|
| + L("~=") |
|
| + L(">") |
|
| + L("<")
|
| +)
|
| +
|
| +MARKER_OP = VERSION_CMP | L("not in") | L("in")
|
| +MARKER_OP.setParseAction(lambda s, l, t: Op(t[0]))
|
| +
|
| +MARKER_VALUE = QuotedString("'") | QuotedString('"')
|
| +MARKER_VALUE.setParseAction(lambda s, l, t: Value(t[0]))
|
| +
|
| +BOOLOP = L("and") | L("or")
|
| +
|
| +MARKER_VAR = VARIABLE | MARKER_VALUE
|
| +
|
| +MARKER_ITEM = Group(MARKER_VAR + MARKER_OP + MARKER_VAR)
|
| +MARKER_ITEM.setParseAction(lambda s, l, t: tuple(t[0]))
|
| +
|
| +LPAREN = L("(").suppress()
|
| +RPAREN = L(")").suppress()
|
| +
|
| +MARKER_EXPR = Forward()
|
| +MARKER_ATOM = MARKER_ITEM | Group(LPAREN + MARKER_EXPR + RPAREN)
|
| +MARKER_EXPR << MARKER_ATOM + ZeroOrMore(BOOLOP + MARKER_EXPR)
|
| +
|
| +MARKER = stringStart + MARKER_EXPR + stringEnd
|
| +
|
| +
|
| +def _coerce_parse_result(results):
|
| + if isinstance(results, ParseResults):
|
| + return [_coerce_parse_result(i) for i in results]
|
| + else:
|
| + return results
|
| +
|
| +
|
| +def _format_marker(marker, first=True):
|
| + assert isinstance(marker, (list, tuple, string_types))
|
| +
|
| + # Sometimes we have a structure like [[...]] which is a single item list
|
| + # where the single item is itself it's own list. In that case we want skip
|
| + # the rest of this function so that we don't get extraneous () on the
|
| + # outside.
|
| + if (isinstance(marker, list) and len(marker) == 1 and
|
| + isinstance(marker[0], (list, tuple))):
|
| + return _format_marker(marker[0])
|
| +
|
| + if isinstance(marker, list):
|
| + inner = (_format_marker(m, first=False) for m in marker)
|
| + if first:
|
| + return " ".join(inner)
|
| + else:
|
| + return "(" + " ".join(inner) + ")"
|
| + elif isinstance(marker, tuple):
|
| + return " ".join([m.serialize() for m in marker])
|
| + else:
|
| + return marker
|
| +
|
| +
|
| +_operators = {
|
| + "in": lambda lhs, rhs: lhs in rhs,
|
| + "not in": lambda lhs, rhs: lhs not in rhs,
|
| + "<": operator.lt,
|
| + "<=": operator.le,
|
| + "==": operator.eq,
|
| + "!=": operator.ne,
|
| + ">=": operator.ge,
|
| + ">": operator.gt,
|
| +}
|
| +
|
| +
|
| +def _eval_op(lhs, op, rhs):
|
| + try:
|
| + spec = Specifier("".join([op.serialize(), rhs]))
|
| + except InvalidSpecifier:
|
| + pass
|
| + else:
|
| + return spec.contains(lhs)
|
| +
|
| + oper = _operators.get(op.serialize())
|
| + if oper is None:
|
| + raise UndefinedComparison(
|
| + "Undefined {0!r} on {1!r} and {2!r}.".format(op, lhs, rhs)
|
| + )
|
| +
|
| + return oper(lhs, rhs)
|
| +
|
| +
|
| +_undefined = object()
|
| +
|
| +
|
| +def _get_env(environment, name):
|
| + value = environment.get(name, _undefined)
|
| +
|
| + if value is _undefined:
|
| + raise UndefinedEnvironmentName(
|
| + "{0!r} does not exist in evaluation environment.".format(name)
|
| + )
|
| +
|
| + return value
|
| +
|
| +
|
| +def _evaluate_markers(markers, environment):
|
| + groups = [[]]
|
| +
|
| + for marker in markers:
|
| + assert isinstance(marker, (list, tuple, string_types))
|
| +
|
| + if isinstance(marker, list):
|
| + groups[-1].append(_evaluate_markers(marker, environment))
|
| + elif isinstance(marker, tuple):
|
| + lhs, op, rhs = marker
|
| +
|
| + if isinstance(lhs, Variable):
|
| + lhs_value = _get_env(environment, lhs.value)
|
| + rhs_value = rhs.value
|
| + else:
|
| + lhs_value = lhs.value
|
| + rhs_value = _get_env(environment, rhs.value)
|
| +
|
| + groups[-1].append(_eval_op(lhs_value, op, rhs_value))
|
| + else:
|
| + assert marker in ["and", "or"]
|
| + if marker == "or":
|
| + groups.append([])
|
| +
|
| + return any(all(item) for item in groups)
|
| +
|
| +
|
| +def format_full_version(info):
|
| + version = '{0.major}.{0.minor}.{0.micro}'.format(info)
|
| + kind = info.releaselevel
|
| + if kind != 'final':
|
| + version += kind[0] + str(info.serial)
|
| + return version
|
| +
|
| +
|
| +def default_environment():
|
| + if hasattr(sys, 'implementation'):
|
| + iver = format_full_version(sys.implementation.version)
|
| + implementation_name = sys.implementation.name
|
| + else:
|
| + iver = '0'
|
| + implementation_name = ''
|
| +
|
| + return {
|
| + "implementation_name": implementation_name,
|
| + "implementation_version": iver,
|
| + "os_name": os.name,
|
| + "platform_machine": platform.machine(),
|
| + "platform_release": platform.release(),
|
| + "platform_system": platform.system(),
|
| + "platform_version": platform.version(),
|
| + "python_full_version": platform.python_version(),
|
| + "platform_python_implementation": platform.python_implementation(),
|
| + "python_version": platform.python_version()[:3],
|
| + "sys_platform": sys.platform,
|
| + }
|
| +
|
| +
|
| +class Marker(object):
|
| +
|
| + def __init__(self, marker):
|
| + try:
|
| + self._markers = _coerce_parse_result(MARKER.parseString(marker))
|
| + except ParseException as e:
|
| + err_str = "Invalid marker: {0!r}, parse error at {1!r}".format(
|
| + marker, marker[e.loc:e.loc + 8])
|
| + raise InvalidMarker(err_str)
|
| +
|
| + def __str__(self):
|
| + return _format_marker(self._markers)
|
| +
|
| + def __repr__(self):
|
| + return "<Marker({0!r})>".format(str(self))
|
| +
|
| + def evaluate(self, environment=None):
|
| + """Evaluate a marker.
|
| +
|
| + Return the boolean from evaluating the given marker against the
|
| + environment. environment is an optional argument to override all or
|
| + part of the determined environment.
|
| +
|
| + The environment is determined from the current Python process.
|
| + """
|
| + current_environment = default_environment()
|
| + if environment is not None:
|
| + current_environment.update(environment)
|
| +
|
| + return _evaluate_markers(self._markers, current_environment)
|
|
|