Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

add support for package format 3 #63

Merged
merged 2 commits into from
Nov 14, 2017
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
37 changes: 35 additions & 2 deletions ament_package/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -123,7 +123,7 @@ def parse_package_string(data, *, filename=None):
"Unable to handle '%s' format version '%d', please update the " \
'manifest file to at least format version 2' % \
(filename, pkg.package_format)
assert pkg.package_format in [2], \
assert pkg.package_format in [2, 3], \
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It would be nice to have some kind of test for this function (parse_package_string()), and have it updated for the new features in package.xml version 3. I guess those didn't make it over from catkin_pkg.

"Unable to handle '%s' format version '%d', please update " \
"'ament_package' (e.g. on Ubuntu/Debian use: sudo apt-get update && " \
'sudo apt-get install --only-upgrade python-ament-package)' % \
Expand All @@ -135,6 +135,8 @@ def parse_package_string(data, *, filename=None):
# version
version_node = _get_node(root, 'version')
pkg.version = _get_node_value(version_node)
pkg.version_compatibility = _get_node_attr(
version_node, 'compatibility', default=None)

# description
pkg.description = _get_node_value(
Expand Down Expand Up @@ -209,6 +211,10 @@ def parse_package_string(data, *, filename=None):
pkg.conflicts = _get_dependencies(root, 'conflict')
pkg.replaces = _get_dependencies(root, 'replace')

# group dependencies and memberships
pkg.group_depends = _get_group_dependencies(root, 'group_depend')
pkg.member_of_groups = _get_group_memberships(root, 'member_of_group')

# exports
export_node = _get_optional_node(root, 'export')
if export_node is not None:
Expand Down Expand Up @@ -242,7 +248,7 @@ def parse_package_string(data, *, filename=None):
]
known = {
'name': [],
'version': [],
'version': ['compatibility'],
'description': [],
'maintainer': ['email'],
'license': [],
Expand All @@ -260,6 +266,11 @@ def parse_package_string(data, *, filename=None):
'replace': depend_attributes,
'export': [],
}
if pkg.package_format > 2:
known.update({
'group_depend': [],
'member_of_group': [],
})
nodes = [n for n in root.childNodes if n.nodeType == n.ELEMENT_NODE]
unknown_tags = {n.tagName for n in nodes if n.tagName not in known.keys()}
if unknown_tags:
Expand Down Expand Up @@ -359,3 +370,25 @@ def _get_dependencies(parent, tagname):
setattr(depend, attr, _get_node_attr(node, attr, default=None))
depends.append(depend)
return depends


def _get_group_dependencies(parent, tagname):
from .group_dependency import GroupDependency
depends = []
for node in _get_nodes(parent, tagname):
depends.append(
GroupDependency(
_get_node_value(node),
condition=_get_node_attr(node, 'condition', default=None)))
return depends


def _get_group_memberships(parent, tagname):
from .group_membership import GroupMembership
memberships = []
for node in _get_nodes(parent, tagname):
memberships.append(
GroupMembership(
_get_node_value(node),
condition=_get_node_attr(node, 'condition', default=None)))
return memberships
76 changes: 76 additions & 0 deletions ament_package/condition.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,76 @@
# Copyright 2014 Open Source Robotics Foundation, 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.

import pyparsing


def evaluate_condition(condition, context):
if condition is None:
return True
expr = _get_condition_expression()
try:
parse_results = expr.parseString(condition, parseAll=True)
except pyparsing.ParseException as e:
raise ValueError(
"condition '%s' failed to parse: %s" % (condition, e))
return _evaluate(parse_results.asList()[0], context)


_condition_expression = None


def _get_condition_expression():
global _condition_expression
if not _condition_expression:
pp = pyparsing
operator = pp.Regex('==|!=').setName('operator')
identifier = pp.Word('$', pp.alphanums + '_', min=2)
value = pp.Word(pp.alphanums + '_-')
comparison_term = identifier | value
condition = pp.Group(comparison_term + operator + comparison_term)
_condition_expression = pp.operatorPrecedence(
condition, [
('and', 2, pp.opAssoc.LEFT, ),
('or', 2, pp.opAssoc.LEFT, ),
])
return _condition_expression


def _evaluate(parse_results, context):
if not isinstance(parse_results, list):
if parse_results.startswith('$'):
# get variable from context
return str(context.get(parse_results[1:], ''))
# return literal value
return parse_results

# recursion
assert len(parse_results) == 3

# handle logical operators
if parse_results[1] == 'and':
return _evaluate(parse_results[0], context) and \
_evaluate(parse_results[2], context)
if parse_results[1] == 'or':
return _evaluate(parse_results[0], context) or \
_evaluate(parse_results[2], context)

# handle comparison operators
assert parse_results[1] in ('==', '!=')
if parse_results[1] == '==':
return _evaluate(parse_results[0], context) == \
_evaluate(parse_results[2], context)
if parse_results[1] == '!=':
return _evaluate(parse_results[0], context) != \
_evaluate(parse_results[2], context)
20 changes: 20 additions & 0 deletions ament_package/dependency.py
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,8 @@
# See the License for the specific language governing permissions and
# limitations under the License.

from ament_package.condition import evaluate_condition


class Dependency:
__slots__ = [
Expand All @@ -21,9 +23,12 @@ class Dependency:
'version_eq',
'version_gte',
'version_gt',
'condition',
'evaluated_condition',
]

def __init__(self, name, **kwargs):
self.evaluated_condition = None
for attr in self.__slots__:
value = kwargs[attr] if attr in kwargs else None
setattr(self, attr, value)
Expand All @@ -41,3 +46,18 @@ def __eq__(self, other):

def __str__(self):
return self.name

def evaluate_condition(self, context):
"""
Evaluate the condition.

The result is also stored in the member variable `evaluated_condition`.

:param context: A dictionary with key value pairs to replace variables
starting with $ in the condition.

:returns: True if the condition evaluates to True, else False
:raises: :exc:`ValueError` if the condition fails to parse
"""
self.evaluated_condition = evaluate_condition(self.condition, context)
return self.evaluated_condition
60 changes: 60 additions & 0 deletions ament_package/group_dependency.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,60 @@
# Copyright 2017 Open Source Robotics Foundation, 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.

from ament_package.condition import evaluate_condition


class GroupDependency:
__slots__ = [
'name',
'condition',
'evaluated_condition',
'members',
]

def __init__(self, name, condition=None, members=None):
self.name = name
self.condition = condition
self.members = members
self.evaluated_condition = None

def __eq__(self, other):
if not isinstance(other, GroupDependency):
return False
return all(getattr(self, attr) == getattr(other, attr)
for attr in self.__slots__)

def __str__(self):
return self.name

def evaluate_condition(self, context):
"""
Evaluate the condition.

The result is also stored in the member variable `evaluated_condition`.

:param context: A dictionary with key value pairs to replace variables
starting with $ in the condition.

:returns: True if the condition evaluates to True, else False
:raises: :exc:`ValueError` if the condition fails to parse
"""
self.evaluated_condition = evaluate_condition(self.condition, context)
return self.evaluated_condition

def extract_group_members(self, packages):
self.members = set()
for pkg in packages:
if self.name in [g.name for g in pkg.member_of_groups]:
self.members.add(pkg.name)
52 changes: 52 additions & 0 deletions ament_package/group_membership.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,52 @@
# Copyright 2017 Open Source Robotics Foundation, 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.

from ament_package.condition import evaluate_condition


class GroupMembership:
__slots__ = [
'name',
'condition',
'evaluated_condition',
]

def __init__(self, name, condition=None):
self.name = name
self.condition = condition
self.evaluated_condition = None

def __eq__(self, other):
if not isinstance(other, GroupMembership):
return False
return all(getattr(self, attr) == getattr(other, attr)
for attr in self.__slots__)

def __str__(self):
return self.name

def evaluate_condition(self, context):
"""
Evaluate the condition.

The result is also stored in the member variable `evaluated_condition`.

:param context: A dictionary with key value pairs to replace variables
starting with $ in the condition.

:returns: True if the condition evaluates to True, else False
:raises: :exc:`ValueError` if the condition fails to parse
"""
self.evaluated_condition = evaluate_condition(self.condition, context)
return self.evaluated_condition
45 changes: 44 additions & 1 deletion ament_package/package.py
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,7 @@ class Package:
'package_format',
'name',
'version',
'version_compatibility',
'description',
'maintainers',
'licenses',
Expand All @@ -39,6 +40,8 @@ class Package:
'doc_depends',
'conflicts',
'replaces',
'group_depends',
'member_of_groups',
'exports',
'filename',
]
Expand Down Expand Up @@ -98,6 +101,32 @@ def get_build_type(self):
return build_type_exports[0]
raise InvalidPackage('Only one <build_type> element is permitted.')

def evaluate_conditions(self, context):
"""
Evaluate the conditions of all dependencies and memberships.

:param context: A dictionary with key value pairs to replace variables
starting with $ in the condition.

:raises: :exc:`ValueError` if any condition fails to parse
"""
for attr in (
'build_depends',
'buildtool_depends',
'build_export_depends',
'buildtool_export_depends',
'exec_depends',
'test_depends',
'doc_depends',
'conflicts',
'replaces',
'group_depends',
'member_of_groups',
):
conditionals = getattr(self, attr)
for conditional in conditionals:
conditional.evaluate_condition(context)

def validate(self):
"""
Ensure that all standards for packages are met.
Expand All @@ -124,11 +153,17 @@ def validate(self):
errors.append("Package name '%s' does not follow naming "
'conventions' % self.name)

version_regexp = '^[0-9]+\.[0-9_]+\.[0-9_]+$'
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Should this be exposed as a constant for sharing with consuming code? So it doesn't necessarily need to be.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It was internal before this patch so I won't consider it part of this PR in order to keep the diff as small as possible.

if not self.version:
errors.append('Package version must not be empty')
elif not re.match('^[0-9]+\.[0-9_]+\.[0-9_]+$', self.version):
elif not re.match(version_regexp, self.version):
errors.append("Package version '%s' does not follow version "
'conventions' % self.version)
if self.version_compatibility:
if not re.match(version_regexp, self.version_compatibility):
errors.append(
"Package compatibility version '%s' does not follow "
'version conventions' % self.version_compatibility)

if not self.description:
errors.append('Package description must not be empty')
Expand Down Expand Up @@ -170,5 +205,13 @@ def validate(self):
"The package must not '%s_depend' on a package with "
'the same name as this package' % dep_type)

if (
{d.name for d in self.group_depends} &
{g.name for g in self.member_of_groups}
):
errors.append(
"The package must not 'group_depend' on a package which it "
'also declares to be a member of')

if errors:
raise InvalidPackage('\n'.join(errors))
Loading