Warning
This document is for an old release of Galaxy. You can alternatively view this page in the latest release if it exists or view the top of the latest release's documentation.
Source code for galaxy.tool_util.linters.general
"""This module contains linting functions for general aspects of the tool."""
import re
import packaging.version
ERROR_VERSION_MSG = "Tool version is missing or empty."
WARN_VERSION_MSG = "Tool version [%s] is not compliant with PEP 440."
VALID_VERSION_MSG = "Tool defines a version [%s]."
ERROR_NAME_MSG = "Tool name is missing or empty."
VALID_NAME_MSG = "Tool defines a name [%s]."
ERROR_ID_MSG = "Tool does not define an id attribute."
VALID_ID_MSG = "Tool defines an id [%s]."
PROFILE_PATTERN = re.compile(r"^[12]\d\.\d{1,2}$")
PROFILE_INFO_DEFAULT_MSG = "Tool targets 16.01 Galaxy profile."
PROFILE_INFO_SPECIFIED_MSG = "Tool specifies profile version [%s]."
PROFILE_INVALID_MSG = "Tool specifies an invalid profile version [%s]."
WARN_WHITESPACE_MSG = "%s contains whitespace, this may cause errors: [%s]."
WARN_WHITESPACE_PRESUFFIX = "%s is pre/suffixed by whitespace, this may cause errors: [%s]."
WARN_ID_WHITESPACE_MSG = "Tool ID contains whitespace - this is discouraged: [%s]."
lint_tool_types = ["*"]
[docs]def lint_general(tool_source, lint_ctx):
"""Check tool version, name, and id."""
# determine line to report for general problems with outputs
tool_xml = getattr(tool_source, "xml_tree", None)
if tool_xml:
tool_node = tool_xml.getroot()
else:
tool_node = None
version = tool_source.parse_version() or ""
parsed_version = packaging.version.parse(version)
if not version:
lint_ctx.error(ERROR_VERSION_MSG, node=tool_node)
elif isinstance(parsed_version, packaging.version.LegacyVersion):
lint_ctx.warn(WARN_VERSION_MSG % version, node=tool_node)
elif version != version.strip():
lint_ctx.warn(WARN_WHITESPACE_PRESUFFIX % ("Tool version", version), node=tool_node)
else:
lint_ctx.valid(VALID_VERSION_MSG % version, node=tool_node)
name = tool_source.parse_name()
if not name:
lint_ctx.error(ERROR_NAME_MSG, node=tool_node)
elif name != name.strip():
lint_ctx.warn(WARN_WHITESPACE_PRESUFFIX % ("Tool name", name), node=tool_node)
else:
lint_ctx.valid(VALID_NAME_MSG % name, node=tool_node)
tool_id = tool_source.parse_id()
if not tool_id:
lint_ctx.error(ERROR_ID_MSG, node=tool_node)
elif re.search(r"\s", tool_id):
lint_ctx.warn(WARN_ID_WHITESPACE_MSG % tool_id, node=tool_node)
else:
lint_ctx.valid(VALID_ID_MSG % tool_id, node=tool_node)
profile = tool_source.parse_profile()
profile_valid = PROFILE_PATTERN.match(profile) is not None
if not profile_valid:
lint_ctx.error(PROFILE_INVALID_MSG % profile, node=tool_node)
elif profile == "16.01":
lint_ctx.valid(PROFILE_INFO_DEFAULT_MSG, node=tool_node)
else:
lint_ctx.valid(PROFILE_INFO_SPECIFIED_MSG % profile, node=tool_node)
requirements, containers, resource_requirements = tool_source.parse_requirements_and_containers()
for r in requirements:
if r.type == "package":
if not r.name:
lint_ctx.error("Requirement without name found")
if not r.version:
lint_ctx.warn(f"Requirement {r.name} defines no version")
# Warn requirement attributes with leading/trailing whitespace:
elif r.version != r.version.strip():
lint_ctx.warn(WARN_WHITESPACE_MSG % ("Requirement version", r.version))
for rr in resource_requirements:
if rr.runtime_required:
lint_ctx.warn("Expressions in resource requirement not supported yet")