import logging
import time
import pytest
import routes
from galaxy import model
from galaxy.app_unittest_utils.toolbox_support import BaseToolBoxTestCase
from galaxy.tool_util.unittest_utils import mock_trans
from galaxy.tool_util.unittest_utils.sample_data import (
SIMPLE_MACRO,
SIMPLE_TOOL_WITH_MACRO,
)
log = logging.getLogger(__name__)
class TestToolBox(BaseToolBoxTestCase):
def test_load_file(self):
self._init_tool()
self._add_config("""""")
toolbox = self.toolbox
assert toolbox.get_tool("test_tool") is not None
assert toolbox.get_tool("not_a_test_tool") is None
def test_record_macros(self):
self._init_tool()
self._init_tool(
filename="tool_with_macro.xml",
tool_contents=SIMPLE_TOOL_WITH_MACRO,
extra_file_contents=SIMPLE_MACRO.substitute(tool_version="2.0"),
extra_file_path="external.xml",
)
self._add_config("""""")
toolbox = self.toolbox
tool = toolbox.get_tool("test_tool")
assert tool is not None
assert len(tool._macro_paths) == 0
tool = toolbox.get_tool("tool_with_macro")
assert tool is not None
assert len(tool._macro_paths) == 1
@pytest.mark.xfail(raises=AssertionError)
def test_tool_reload_when_macro_is_altered(self):
self._init_tool(
filename="tool_with_macro.xml",
tool_contents=SIMPLE_TOOL_WITH_MACRO,
extra_file_contents=SIMPLE_MACRO.substitute(tool_version="2.0"),
extra_file_path="external.xml",
)
self._add_config("""""")
toolbox = self.toolbox
tool = toolbox.get_tool("tool_with_macro")
assert tool is not None
assert len(tool._macro_paths) == 1
macro_path = tool._macro_paths[0]
with open(macro_path, "w") as macro_out:
macro_out.write(SIMPLE_MACRO.substitute(tool_version="3.0"))
def check_tool_macro():
tool = self.toolbox.get_tool("tool_with_macro")
assert tool.version == "3.0"
self._try_until_no_errors(check_tool_macro)
@pytest.mark.xfail(raises=AssertionError)
def test_tool_reload_for_broken_tool(self):
self._init_tool(filename="simple_tool.xml", version="1.0")
self._add_config("""""")
toolbox = self.toolbox
tool = toolbox.get_tool("test_tool")
assert tool is not None
assert tool.version == "1.0"
assert tool.tool_errors is None
# Tool is loaded, now let's break it
tool_path = tool.config_file
with open(tool.config_file, "w") as out:
out.write("certainly not a valid tool")
def check_tool_errors():
tool = self.toolbox.get_tool("test_tool")
assert tool is not None
assert tool.version == "1.0"
assert tool.tool_errors == "Current on-disk tool is not valid"
self._try_until_no_errors(check_tool_errors)
# Tool is still loaded, lets restore it with a new version
self._init_tool(filename="simple_tool.xml", version="2.0")
def check_no_tool_errors():
tool = self.toolbox.get_tool("test_tool")
assert tool is not None
assert tool.version == "2.0"
assert tool.tool_errors is None
assert tool_path == tool.config_file, "config file"
self._try_until_no_errors(check_no_tool_errors)
def _try_until_no_errors(self, f):
e = None
for _ in range(40):
try:
f()
return
except AssertionError as error:
e = error
time.sleep(0.25)
if e:
raise e
def test_enforce_tool_profile(self):
self._init_tool(filename="old_tool.xml", version="1.0", profile="17.01", tool_id="test_old_tool_profile")
with self.assertRaisesRegex(Exception, r"The tool \[test_new_tool_profile\] targets version 37\.01 of Galaxy"):
# This will write the file but fail to load the tool
self._init_tool(filename="new_tool.xml", version="2.0", profile="37.01", tool_id="test_new_tool_profile")
self._add_config("""""")
toolbox = self.toolbox
assert toolbox.get_tool("test_old_tool_profile") is not None
assert toolbox.get_tool("test_new_tool_profile") is None
def test_to_dict_in_panel(self):
for json_conf in [True, False]:
self._init_tool_in_section(json=json_conf)
mapper = routes.Mapper()
mapper.connect("tool_runner", "/test/tool_runner")
as_dict = self.toolbox.to_dict(mock_trans())
test_section = self._find_section(as_dict, "t")
assert len(test_section["elems"]) == 1
assert test_section["elems"][0]["id"] == "test_tool"
def test_to_dict_out_of_panel(self):
for json_conf in [True, False]:
self._init_tool_in_section(json=json_conf)
mapper = routes.Mapper()
mapper.connect("tool_runner", "/test/tool_runner")
as_dict = self.toolbox.to_dict(mock_trans(), in_panel=False)
assert as_dict[0]["id"] == "test_tool"
def test_out_of_panel_filtering(self):
self._init_tool_in_section()
mapper = routes.Mapper()
mapper.connect("tool_runner", "/test/tool_runner")
as_dict = self.toolbox.to_dict(mock_trans(), in_panel=False)
assert len(as_dict) == 1
def allow_user_access(user, attempting_access):
assert not attempting_access
return False
# Disable access to the tool, make sure it is filtered out.
self.toolbox.get_tool("test_tool").allow_user_access = allow_user_access
as_dict = self.toolbox.to_dict(mock_trans(), in_panel=False)
assert len(as_dict) == 0, as_dict
def _find_section(self, as_dict, section_id):
for elem in as_dict:
if elem.get("id") == section_id:
assert elem["model_class"] == "ToolSection"
return elem
raise AssertionError(f"Failed to find section with id [{section_id}]")
def test_tool_shed_properties(self):
self._init_tool()
self._setup_two_versions_in_config(section=False)
self._setup_two_versions()
test_tool = self.toolbox.get_tool("test_tool")
assert test_tool.tool_shed == "github.com"
assert test_tool.repository_owner == "galaxyproject"
assert test_tool.repository_name == "example"
# TODO: Not deterministc, probably should be?
assert test_tool.installed_changeset_revision in ["1", "2"]
def test_tool_shed_properties_only_on_installed_tools(self):
self._init_tool()
self._add_config("""""")
toolbox = self.toolbox
test_tool = toolbox.get_tool("test_tool")
assert test_tool.tool_shed is None
assert test_tool.repository_name is None
assert test_tool.repository_owner is None
assert test_tool.installed_changeset_revision is None
def test_tool_shed_request_version(self):
self._init_tool()
self._setup_two_versions_in_config(section=False)
self._setup_two_versions()
test_tool = self.toolbox.get_tool("test_tool", tool_version="0.1")
assert test_tool.version == "0.1"
test_tool = self.toolbox.get_tool("test_tool", tool_version="0.2")
assert test_tool.version == "0.2"
# there is no version 3, return newest version
test_tool = self.toolbox.get_tool("test_tool", tool_version="3")
assert test_tool.version == "0.2"
def test_load_file_in_section(self):
self._init_tool_in_section()
toolbox = self.toolbox
assert toolbox.get_tool("test_tool") is not None
assert toolbox.get_tool("not_a_test_tool") is None
def test_writes_integrate_tool_panel(self):
self._init_tool()
self._add_config("""""")
self.assert_integerated_tool_panel(exists=False)
self.toolbox # noqa: B018
self.assert_integerated_tool_panel(exists=True)
def test_groups_tools_in_section(self):
self._init_tool()
self._setup_two_versions_in_config(section=True)
self._setup_two_versions()
self.toolbox # noqa: B018
self.__verify_two_test_tools()
# Assert only newer version of the tool loaded into the panel.
section = self.toolbox._tool_panel["tid"]
assert len(section.elems) == 1
assert next(iter(section.elems.values())).id == "github.com/galaxyproject/example/test_tool/0.2"
test_tool = self.toolbox.get_tool("test_tool", tool_version="0.1")
section_pair = self.toolbox.get_section_for_tool(test_tool)
assert section_pair == ("tid", "TID")
def test_group_tools_out_of_section(self):
self._init_tool()
self._setup_two_versions_in_config(section=False)
self._setup_two_versions()
self.__verify_two_test_tools()
# Assert tools merged in tool panel.
assert len(self.toolbox._tool_panel) == 2 # 1 tool (w 2 versions) + built-in converters
def test_get_section_by_label(self):
self._add_config(
""""""
)
print(self.toolbox._tool_panel)
assert len(self.toolbox._tool_panel) == 2 # section + built-in converters
section = self.toolbox._tool_panel["tid"]
tool_panel_section_key, section_by_label = self.toolbox.get_section(
section_id="nope", new_label="Completely unrelated", create_if_needed=True
)
assert section_by_label is section
assert tool_panel_section_key == "tid"
def test_get_tool(self):
self._init_tool()
self._setup_two_versions_in_config()
self._setup_two_versions()
assert self.toolbox.get_tool("test_tool").id in [
"github.com/galaxyproject/example/test_tool/0.1",
"github.com/galaxyproject/example/test_tool/0.2",
]
assert (
self.toolbox.get_tool("github.com/galaxyproject/example/test_tool/0.1").id
== "github.com/galaxyproject/example/test_tool/0.1"
)
assert (
self.toolbox.get_tool("github.com/galaxyproject/example/test_tool/0.2").id
== "github.com/galaxyproject/example/test_tool/0.2"
)
assert (
self.toolbox.get_tool("github.com/galaxyproject/example/test_tool/0.3").id
!= "github.com/galaxyproject/example/test_tool/0.3"
)
def test_tool_dir(self):
self._init_tool()
self._add_config(f"""""")
toolbox = self.toolbox
assert toolbox.get_tool("test_tool") is not None
def test_tool_dir_json(self):
self._init_tool()
self._add_config({"items": [{"type": "tool_dir", "dir": self.test_directory}]}, name="tool_conf.json")
toolbox = self.toolbox
assert toolbox.get_tool("test_tool") is not None
def test_workflow_in_panel(self):
stored_workflow = self.__test_workflow()
encoded_id = self.app.security.encode_id(stored_workflow.id)
self._add_config(f"""""")
assert len(self.toolbox._tool_panel) == 2 # workflow + built-in converters section
panel_workflow = next(iter(self.toolbox._tool_panel.values()))
assert panel_workflow == stored_workflow.latest_workflow
# TODO: test to_dict with workflows
def test_workflow_in_section(self):
stored_workflow = self.__test_workflow()
encoded_id = self.app.security.encode_id(stored_workflow.id)
self._add_config(
f""""""
)
assert len(self.toolbox._tool_panel) == 2 # workflow + built-in converters section
section = self.toolbox._tool_panel["tid"]
assert len(section.elems) == 1
panel_workflow = next(iter(section.elems.values()))
assert panel_workflow == stored_workflow.latest_workflow
def test_label_in_panel(self):
self._add_config("""""")
assert len(self.toolbox._tool_panel) == 3 # 2 labels + built-in converters section
self.__check_test_labels(self.toolbox._tool_panel)
def test_label_in_section(self):
self._add_config(
""""""
)
assert len(self.toolbox._tool_panel) == 2 # section + built-in converters section
section = self.toolbox._tool_panel["tid"]
self.__check_test_labels(section.elems)
def _init_tool_in_section(self, json=False):
self._init_tool()
if not json:
self._add_config("""""")
else:
section = {
"type": "section",
"id": "t",
"name": "test",
"items": [{"type": "tool", "file": "tool.xml"}],
}
self._add_config({"items": [section]}, name="tool_conf.json")
def __check_test_labels(self, panel_dict):
# if panel_dict is the complete panel it will contain builtin_converters
# if its a section then not
assert list(panel_dict.keys()) in (
["label_lab1", "label_lab2", "builtin_converters"],
["label_lab1", "label_lab2"],
)
label1 = next(iter(panel_dict.values()))
assert label1.id == "lab1"
assert label1.text == "Label 1"
label2 = panel_dict["label_lab2"]
assert label2.id == "lab2"
assert label2.text == "Label 2"
def __test_workflow(self):
stored_workflow = model.StoredWorkflow()
workflow = model.Workflow()
workflow.stored_workflow = stored_workflow
stored_workflow.latest_workflow = workflow
user = model.User()
user.email = "test@example.com"
user.password = "passw0rD1"
stored_workflow.user = user
session = self.app.model.context
session.add(workflow)
session.add(stored_workflow)
session.commit()
return stored_workflow
def __verify_two_test_tools(self):
# Assert tool versions of the tool with simple id 'test_tool'
all_versions = self.toolbox.get_tool("test_tool", get_all_versions=True)
assert len(all_versions) == 2
# Verify lineage_ids on both tools is correctly ordered.
for version in ["0.1", "0.2"]:
guid = "github.com/galaxyproject/example/test_tool/" + version
lineage_ids = self.toolbox.get_tool(guid).lineage.get_version_ids()
assert lineage_ids[0] == "github.com/galaxyproject/example/test_tool/0.1"
assert lineage_ids[1] == "github.com/galaxyproject/example/test_tool/0.2"
# Test tool_version attribute.
assert (
self.toolbox.get_tool("test_tool", tool_version="0.1").guid
== "github.com/galaxyproject/example/test_tool/0.1"
)
assert (
self.toolbox.get_tool("test_tool", tool_version="0.2").guid
== "github.com/galaxyproject/example/test_tool/0.2"
)
def test_builtin_converters_in_integrated_panel(self):
self._init_tool()
self._add_config("""""")
toolbox = self.toolbox
assert "builtin_converters" in toolbox._tool_panel
assert "builtin_converters" in toolbox._integrated_tool_panel
def test_default_lineage(self):
self.__init_versioned_tools()
self._add_config("""""")
self.__verify_get_tool_for_default_lineage()
def test_default_lineage_reversed(self):
# Run same test as above but with entries in tool_conf reversed to
# ensure versioning is at work and not order effects.
self.__init_versioned_tools()
self._add_config("""""")
self.__verify_get_tool_for_default_lineage()
def test_grouping_with_default_lineage(self):
self.__init_versioned_tools()
self._add_config("""""")
self.__verify_tool_panel_for_default_lineage()
def test_grouping_with_default_lineage_reversed(self):
# Run same test as above but with entries in tool_conf reversed to
# ensure versioning is at work and not order effects.
self.__init_versioned_tools()
self._add_config("""""")
self.__verify_tool_panel_for_default_lineage()
def __init_versioned_tools(self):
self._init_tool(filename="tool_v01.xml", version="0.1")
self._init_tool(filename="tool_v02.xml", version="0.2")
def __verify_tool_panel_for_default_lineage(self):
assert len(self.toolbox._tool_panel) == 2 # tool + built-in-converters section
tool = self.toolbox._tool_panel["tool_test_tool"]
assert tool.version == "0.2", tool.version
assert tool.id == "test_tool"
def __verify_get_tool_for_default_lineage(self):
tool_v01 = self.toolbox.get_tool("test_tool", tool_version="0.1")
tool_v02 = self.toolbox.get_tool("test_tool", tool_version="0.2")
assert tool_v02.id == "test_tool"
assert tool_v02.version == "0.2", tool_v02.version
assert tool_v01.id == "test_tool"
assert tool_v01.version == "0.1"
# Newer variant gets to be default for that id.
default_tool = self.toolbox.get_tool("test_tool")
assert default_tool.id == "test_tool"
assert default_tool.version == "0.2"