Skip to content

Commit 7016f8f

Browse files
committed
Refactor graph tests
- use pytest functions which can use fixtures - use fixtures to reduce network roundtrips - use parametrized fixture to control igraph usage - also splits igraph tests into 2
1 parent 647d688 commit 7016f8f

File tree

3 files changed

+145
-98
lines changed

3 files changed

+145
-98
lines changed

pymaid/tests/conftest.py

+63
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,63 @@
1+
import os
2+
import pytest
3+
import navis
4+
from contextlib import contextmanager
5+
import logging
6+
import pymaid
7+
8+
logger = logging.getLogger(__name__)
9+
10+
11+
@contextmanager
12+
def igraph_context(use_igraph):
13+
orig = navis.config.use_igraph
14+
logger.debug(f"Setting navis.config.use_igraph = {use_igraph}")
15+
navis.config.use_igraph = use_igraph
16+
yield use_igraph
17+
logger.debug(f"Resetting navis.config.use_igraph = {orig}")
18+
navis.config.use_igraph = orig
19+
20+
21+
@pytest.fixture(scope="session", params=[True, False])
22+
def use_igraph(request):
23+
with igraph_context(request.param) as use:
24+
yield use
25+
26+
27+
@pytest.fixture(scope="module")
28+
def client():
29+
return pymaid.CatmaidInstance(
30+
os.environ.get("PYMAID_TEST_SERVER_URL", 'https://fafb.catmaid.virtualflybrain.org/'),
31+
os.environ.get("PYMAID_TEST_TOKEN"),
32+
os.environ.get("PYMAID_TEST_HTTP_USER"),
33+
os.environ.get("PYMAID_TEST_HTTP_PW"),
34+
make_global=True,
35+
)
36+
37+
38+
@pytest.fixture(scope="module")
39+
def skids():
40+
return [
41+
int(s) for s in os.environ.get(
42+
"PYMAID_TEST_SKIDS",
43+
"16,1299740,4744251"
44+
).split(",")
45+
]
46+
47+
48+
@pytest.fixture(scope="module")
49+
def annotation_names():
50+
return os.environ.get(
51+
"PYMAID_TEST_ANNOTATIONS",
52+
'Paper: Dolan and Belliart-Guérin et al. 2018,Paper: Wang et al 2020a'
53+
).split(",")
54+
55+
56+
@pytest.fixture(scope="module")
57+
def volume_name():
58+
return os.environ.get("PYMAID_TEST_VOLUME", "LH_R")
59+
60+
61+
@pytest.fixture(scope="module")
62+
def stack_id():
63+
return int(os.environ.get("PYMAID_TEST_STACK_ID", 1))

pymaid/tests/test_graph.py

+81
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,81 @@
1+
import pytest
2+
import pymaid
3+
import navis as ns
4+
import numpy as np
5+
6+
SEED = 1991
7+
8+
@pytest.fixture(scope="module")
9+
def neuron_list(client, skids):
10+
return pymaid.get_neuron(skids[0:2], remote_instance=client)
11+
12+
13+
@pytest.fixture(scope="module")
14+
def neuron(neuron_list):
15+
n = neuron_list[0]
16+
return n.reroot(n.soma, inplace=False)
17+
18+
@pytest.fixture(scope="module")
19+
def leaf_slab(neuron):
20+
rng = np.random.default_rng(SEED)
21+
leaf_id = neuron.nodes[neuron.nodes.type == 'end'].sample(
22+
1, random_state=rng).iloc[0].node_id
23+
slab_id = neuron.nodes[neuron.nodes.type == 'slab'].sample(
24+
1, random_state=rng).iloc[0].node_id
25+
return (leaf_id, slab_id)
26+
27+
28+
def test_reroot(neuron_list, neuron, leaf_slab, use_igraph):
29+
leaf_id, slab_id = leaf_slab
30+
assert neuron.reroot(leaf_id, inplace=False) is not None
31+
assert neuron_list.reroot(neuron_list.soma, inplace=False) is not None
32+
33+
34+
def test_distal_to(neuron, leaf_slab, use_igraph):
35+
leaf_id, slab_id = leaf_slab
36+
assert ns.distal_to(neuron, leaf_id, neuron.root)
37+
assert not ns.distal_to(neuron, neuron.root, leaf_id)
38+
39+
40+
def test_distance(neuron, use_igraph):
41+
leaf_id = neuron.nodes[neuron.nodes.type == 'end'].iloc[0].node_id
42+
43+
assert ns.dist_between(neuron, leaf_id, neuron.root) is not None
44+
assert ns.dist_between(neuron, neuron.root, leaf_id) is not None
45+
46+
47+
def test_find_bp(neuron, use_igraph):
48+
assert ns.find_main_branchpoint(neuron, reroot_soma=False) is not None
49+
50+
51+
def test_split_fragments(neuron, use_igraph):
52+
assert ns.split_into_fragments(neuron, n=2, reroot_soma=False) is not None
53+
54+
55+
def test_longest_neurite(neuron, use_igraph):
56+
assert ns.longest_neurite(neuron, n=2, reroot_soma=False) is not None
57+
58+
59+
def test_cut_neuron(neuron, leaf_slab, use_igraph):
60+
leaf_id, slab_id = leaf_slab
61+
dist, prox = ns.cut_skeleton(neuron, slab_id)
62+
assert dist.nodes.shape != prox.nodes.shape
63+
64+
# Make sure dist and prox check out
65+
assert ns.distal_to(neuron, dist.root, prox.root)
66+
67+
68+
def test_subset(neuron, use_igraph):
69+
assert isinstance(
70+
ns.subset_neuron(neuron, neuron.segments[0]),
71+
pymaid.CatmaidNeuron
72+
)
73+
74+
75+
def test_node_sorting(neuron, use_igraph):
76+
result = ns.graph.node_label_sorting(neuron)
77+
assert isinstance(result, np.ndarray)
78+
79+
80+
def test_geodesic_matrix(neuron, use_igraph):
81+
geo = ns.geodesic_matrix(neuron)

pymaid/tests/test_pymaid.py

+1-98
Original file line numberDiff line numberDiff line change
@@ -35,6 +35,7 @@
3535
import warnings
3636
import os
3737
import matplotlib as mpl
38+
import pytest
3839
#if os.environ.get('DISPLAY', '') == '':
3940
# warnings.warn('No display found. Using template backend (nothing '
4041
# 'will show).')
@@ -605,104 +606,6 @@ def test_guess_radius(self):
605606
pymaid.CatmaidNeuron)
606607

607608

608-
class TestGraphs(unittest.TestCase):
609-
"""Test graph functions."""
610-
611-
def try_conditions(func):
612-
"""Runs each test under various conditions and asserts that results
613-
are always the same."""
614-
615-
def wrapper(self, *args, **kwargs):
616-
ns.config.use_igraph = False
617-
res1 = func(self, *args, **kwargs)
618-
if igraph:
619-
ns.config.use_igraph = True
620-
res2 = func(self, *args, **kwargs)
621-
self.assertEqual(res1, res2)
622-
return res1
623-
return wrapper
624-
625-
def setUp(self):
626-
self.rm = pymaid.CatmaidInstance(server=config_test.server_url,
627-
http_user=config_test.http_user,
628-
http_password=config_test.http_pw,
629-
api_token=config_test.token,
630-
make_global=True)
631-
632-
self.nl = pymaid.get_neuron(config_test.test_skids[0:2],
633-
remote_instance=self.rm)
634-
635-
self.n = self.nl[0]
636-
self.n.reroot(self.n.soma)
637-
638-
# Get some random leaf node
639-
self.leaf_id = self.n.nodes[self.n.nodes.type == 'end'].sample(
640-
1).iloc[0].node_id
641-
self.slab_id = self.n.nodes[self.n.nodes.type == 'slab'].sample(
642-
1).iloc[0].node_id
643-
644-
@try_conditions
645-
def test_reroot(self):
646-
self.assertIsNotNone(self.n.reroot(self.leaf_id, inplace=False))
647-
self.assertIsNotNone(self.nl.reroot(self.nl.soma, inplace=False))
648-
649-
@try_conditions
650-
def test_distal_to(self):
651-
self.assertTrue(ns.distal_to(self.n, self.leaf_id, self.n.root))
652-
self.assertFalse(ns.distal_to(self.n, self.n.root, self.leaf_id))
653-
654-
@try_conditions
655-
def test_distance(self):
656-
leaf_id = self.n.nodes[self.n.nodes.type == 'end'].iloc[0].node_id
657-
658-
self.assertIsNotNone(ns.dist_between(self.n,
659-
leaf_id,
660-
self.n.root))
661-
self.assertIsNotNone(ns.dist_between(self.n,
662-
self.n.root,
663-
leaf_id))
664-
665-
@try_conditions
666-
def test_find_bp(self):
667-
self.assertIsNotNone(ns.find_main_branchpoint(self.n,
668-
reroot_soma=False))
669-
670-
@try_conditions
671-
def test_split_fragments(self):
672-
self.assertIsNotNone(ns.split_into_fragments(self.n,
673-
n=2,
674-
reroot_soma=False))
675-
676-
@try_conditions
677-
def test_longest_neurite(self):
678-
self.assertIsNotNone(ns.longest_neurite(self.n,
679-
n=2,
680-
reroot_soma=False))
681-
682-
@try_conditions
683-
def test_cut_neuron(self):
684-
dist, prox = ns.cut_skeleton(self.n, self.slab_id)
685-
self.assertNotEqual(dist.nodes.shape, prox.nodes.shape)
686-
687-
# Make sure dist and prox check out
688-
self.assertTrue(ns.distal_to(self.n, dist.root, prox.root))
689-
690-
@try_conditions
691-
def test_subset(self):
692-
self.assertIsInstance(ns.subset_neuron(self.n,
693-
self.n.segments[0]),
694-
pymaid.CatmaidNeuron)
695-
696-
@try_conditions
697-
def test_node_sorting(self):
698-
self.assertIsInstance(ns.graph.node_label_sorting(self.n),
699-
list)
700-
701-
@try_conditions
702-
def test_geodesic_matrix(self):
703-
geo = ns.geodesic_matrix(self.n)
704-
705-
706609
class TestConnectivity(unittest.TestCase):
707610
"""Test connectivity-related functions."""
708611

0 commit comments

Comments
 (0)