From ef1b4110acc97c81c6105ebd426e0698444ba358 Mon Sep 17 00:00:00 2001 From: Matt McCormick Date: Mon, 22 Apr 2024 13:36:13 -0400 Subject: [PATCH 1/7] ENH: Migrate from flit to hatch More features. --- src/itk_dreg/__init__.py | 4 ++-- src/pyproject.toml | 29 ++++++++++++++++++----------- 2 files changed, 20 insertions(+), 13 deletions(-) diff --git a/src/itk_dreg/__init__.py b/src/itk_dreg/__init__.py index 238e73a..2728588 100644 --- a/src/itk_dreg/__init__.py +++ b/src/itk_dreg/__init__.py @@ -1,5 +1,5 @@ """ -ITK-DREG Distributed registration framework. +itk-dreg distributed registration framework. """ -__version__ = "0.0.1" +__version__ = "0.1.0" diff --git a/src/pyproject.toml b/src/pyproject.toml index db2a241..2568f83 100644 --- a/src/pyproject.toml +++ b/src/pyproject.toml @@ -1,10 +1,13 @@ [build-system] -requires = ["flit_core >=3.2,<4"] -build-backend = "flit_core.buildapi" +requires = ["hatchling"] +build-backend = "hatchling.build" [project] -name = "itk_dreg" -authors = [{name = "InsightSoftwareConsortium", email = "matt.mccormick@kitware.com"}] +name = "itk-dreg" +authors = [ + {name = "Matt McCormick", email = "matt.mccormick@kitware.com"}, + {name = "Tom Birdsong", email = "tom.birdsong@kitware.com"} +] license = {file = "LICENSE"} classifiers = [ "License :: OSI Approved :: Apache Software License", @@ -26,23 +29,25 @@ classifiers = [ "Operating System :: MacOS" ] requires-python = ">=3.9" -dynamic = ["version", "description"] +dynamic = ["version"] readme = "../README.md" -keywords = ['ITK','InsightToolkit'] +keywords = ['itk','InsightToolkit','registration', 'brain','distributed', 'dask'] dependencies = [ 'dask[distributed] >=2023.10.0', - 'itk >=5.4rc02', + 'itk >=5.4rc04', 'numpy' ] [project.urls] -Home = "https://github.com/InsightSoftwareConsortium/itk-dreg" +Home = "https://itk-dreg.readthedocs.io/" +Source = "https://github.com/InsightSoftwareConsortium/itk-dreg" +Issues = "https://github.com/InsightSoftwareConsortium/issues" [project.optional-dependencies] test = [ - 'itk-elastix>=0.19.1', - 'itk-ioomezarrngff>=0.2.1', + 'itk-elastix>=0.19.2', + 'itk-ioomezarrngff>=0.3rc1', 'nbmake', 'pytest', 'scipy>=1.11.3' @@ -60,7 +65,7 @@ doc = [ ] impl = [ - 'itk-elastix>=0.19.0', + 'itk-elastix>=0.19.2', 'scipy>=1.11.3' ] @@ -80,3 +85,5 @@ notebook = [ [tool.ruff] extend-ignore = ["E501"] +[tool.hatch.version] +path = "itk_dreg/__init__.py" \ No newline at end of file From e49d16444b9a193211995384e8f938541fbb126b Mon Sep 17 00:00:00 2001 From: Matt McCormick Date: Mon, 22 Apr 2024 14:05:45 -0400 Subject: [PATCH 2/7] STYLE: Move test/ directory next to package Per Python conventions. --- .github/workflows/build-test-publish.yml | 10 ++++++---- src/pyproject.toml | 16 +++++++++++++++- {test => src/test}/elastix/test_elx_import.py | 0 .../test}/elastix/test_elx_integration.py | 0 {test => src/test}/elastix/test_elx_serialize.py | 0 .../test}/itk_dreg/test_block_interface.py | 0 .../test}/itk_dreg/test_dreg_block_subdivide.py | 0 {test => src/test}/itk_dreg/test_import.py | 0 {test => src/test}/itk_dreg/test_scheduler.py | 0 {test => src/test}/itk_dreg/test_serialize.py | 0 .../test}/reduce_dfield/test_integration.py | 0 .../test}/reduce_dfield/test_reduce_dfield.py | 0 .../reduce_dfield/test_transform_collection.py | 0 {test => src/test}/util/__init__.py | 0 {test => src/test}/util/mock.py | 0 15 files changed, 21 insertions(+), 5 deletions(-) rename {test => src/test}/elastix/test_elx_import.py (100%) rename {test => src/test}/elastix/test_elx_integration.py (100%) rename {test => src/test}/elastix/test_elx_serialize.py (100%) rename {test => src/test}/itk_dreg/test_block_interface.py (100%) rename {test => src/test}/itk_dreg/test_dreg_block_subdivide.py (100%) rename {test => src/test}/itk_dreg/test_import.py (100%) rename {test => src/test}/itk_dreg/test_scheduler.py (100%) rename {test => src/test}/itk_dreg/test_serialize.py (100%) rename {test => src/test}/reduce_dfield/test_integration.py (100%) rename {test => src/test}/reduce_dfield/test_reduce_dfield.py (100%) rename {test => src/test}/reduce_dfield/test_transform_collection.py (100%) rename {test => src/test}/util/__init__.py (100%) rename {test => src/test}/util/mock.py (100%) diff --git a/.github/workflows/build-test-publish.yml b/.github/workflows/build-test-publish.yml index 4c4b2b9..5a1283e 100644 --- a/.github/workflows/build-test-publish.yml +++ b/.github/workflows/build-test-publish.yml @@ -37,16 +37,16 @@ jobs: - name: "Install Dependencies" shell: bash run: | - python -m pip install flit + python -m pip install hatch cd src - flit install --deps develop + hatch build - name: Download Python Wheel Artifact uses: actions/download-artifact@v4 with: name: wheel - - name: "Install itk_dreg from Wheel Artifact" + - name: "Install itk-dreg from Wheel Artifact" run: | wheel_name_full=`(find . -name "itk_dreg-*.whl")` python -m pip uninstall -y itk_dreg @@ -55,7 +55,9 @@ jobs: - name: "Run Tests" shell: bash run: | - pytest ./test -vvv -s -k "not localcluster and not serialize_pairwise_result" + cd src + pip install -e '.[test]' + pytest -vvv -s -k "not localcluster and not serialize_pairwise_result" publish: needs: diff --git a/src/pyproject.toml b/src/pyproject.toml index 2568f83..12e6d38 100644 --- a/src/pyproject.toml +++ b/src/pyproject.toml @@ -86,4 +86,18 @@ notebook = [ extend-ignore = ["E501"] [tool.hatch.version] -path = "itk_dreg/__init__.py" \ No newline at end of file +path = "itk_dreg/__init__.py" + +[tool.hatch.envs.default] +dependencies = [ + 'itk-elastix>=0.19.2', + 'scipy>=1.11.3', + 'itk-ioomezarrngff>=0.3rc1', + 'nbmake', + 'pytest', +] + +[tool.hatch.envs.default.scripts] +test = [ + "pytest -vvv -s -k \"not localcluster and not serialize_pairwise_result\"" +] diff --git a/test/elastix/test_elx_import.py b/src/test/elastix/test_elx_import.py similarity index 100% rename from test/elastix/test_elx_import.py rename to src/test/elastix/test_elx_import.py diff --git a/test/elastix/test_elx_integration.py b/src/test/elastix/test_elx_integration.py similarity index 100% rename from test/elastix/test_elx_integration.py rename to src/test/elastix/test_elx_integration.py diff --git a/test/elastix/test_elx_serialize.py b/src/test/elastix/test_elx_serialize.py similarity index 100% rename from test/elastix/test_elx_serialize.py rename to src/test/elastix/test_elx_serialize.py diff --git a/test/itk_dreg/test_block_interface.py b/src/test/itk_dreg/test_block_interface.py similarity index 100% rename from test/itk_dreg/test_block_interface.py rename to src/test/itk_dreg/test_block_interface.py diff --git a/test/itk_dreg/test_dreg_block_subdivide.py b/src/test/itk_dreg/test_dreg_block_subdivide.py similarity index 100% rename from test/itk_dreg/test_dreg_block_subdivide.py rename to src/test/itk_dreg/test_dreg_block_subdivide.py diff --git a/test/itk_dreg/test_import.py b/src/test/itk_dreg/test_import.py similarity index 100% rename from test/itk_dreg/test_import.py rename to src/test/itk_dreg/test_import.py diff --git a/test/itk_dreg/test_scheduler.py b/src/test/itk_dreg/test_scheduler.py similarity index 100% rename from test/itk_dreg/test_scheduler.py rename to src/test/itk_dreg/test_scheduler.py diff --git a/test/itk_dreg/test_serialize.py b/src/test/itk_dreg/test_serialize.py similarity index 100% rename from test/itk_dreg/test_serialize.py rename to src/test/itk_dreg/test_serialize.py diff --git a/test/reduce_dfield/test_integration.py b/src/test/reduce_dfield/test_integration.py similarity index 100% rename from test/reduce_dfield/test_integration.py rename to src/test/reduce_dfield/test_integration.py diff --git a/test/reduce_dfield/test_reduce_dfield.py b/src/test/reduce_dfield/test_reduce_dfield.py similarity index 100% rename from test/reduce_dfield/test_reduce_dfield.py rename to src/test/reduce_dfield/test_reduce_dfield.py diff --git a/test/reduce_dfield/test_transform_collection.py b/src/test/reduce_dfield/test_transform_collection.py similarity index 100% rename from test/reduce_dfield/test_transform_collection.py rename to src/test/reduce_dfield/test_transform_collection.py diff --git a/test/util/__init__.py b/src/test/util/__init__.py similarity index 100% rename from test/util/__init__.py rename to src/test/util/__init__.py diff --git a/test/util/mock.py b/src/test/util/mock.py similarity index 100% rename from test/util/mock.py rename to src/test/util/mock.py From e26de24eb407f0c00cdcc599429129609ca4ceee Mon Sep 17 00:00:00 2001 From: Matt McCormick Date: Mon, 22 Apr 2024 14:10:15 -0400 Subject: [PATCH 3/7] STYLE: ruff --fix --- src/pyproject.toml | 2 +- src/test/elastix/test_elx_import.py | 4 +--- src/test/elastix/test_elx_integration.py | 1 - src/test/elastix/test_elx_serialize.py | 2 -- src/test/itk_dreg/test_import.py | 9 +-------- src/test/itk_dreg/test_scheduler.py | 1 - src/test/reduce_dfield/test_integration.py | 1 - src/test/reduce_dfield/test_reduce_dfield.py | 6 +----- src/test/util/mock.py | 4 +--- 9 files changed, 5 insertions(+), 25 deletions(-) diff --git a/src/pyproject.toml b/src/pyproject.toml index 12e6d38..80237ca 100644 --- a/src/pyproject.toml +++ b/src/pyproject.toml @@ -99,5 +99,5 @@ dependencies = [ [tool.hatch.envs.default.scripts] test = [ - "pytest -vvv -s -k \"not localcluster and not serialize_pairwise_result\"" + "pytest -vvv -s" ] diff --git a/src/test/elastix/test_elx_import.py b/src/test/elastix/test_elx_import.py index 73035d1..5388f08 100644 --- a/src/test/elastix/test_elx_import.py +++ b/src/test/elastix/test_elx_import.py @@ -12,6 +12,4 @@ def test_loadmodule(): - import itk_dreg.elastix - import itk_dreg.elastix.util - import itk_dreg.elastix.register + pass diff --git a/src/test/elastix/test_elx_integration.py b/src/test/elastix/test_elx_integration.py index 6e6e2fe..588ec9f 100644 --- a/src/test/elastix/test_elx_integration.py +++ b/src/test/elastix/test_elx_integration.py @@ -7,7 +7,6 @@ import itk import numpy as np import dask -import dask.array as da sys.path.append('./src') import itk_dreg.itk diff --git a/src/test/elastix/test_elx_serialize.py b/src/test/elastix/test_elx_serialize.py index e3de240..a836642 100644 --- a/src/test/elastix/test_elx_serialize.py +++ b/src/test/elastix/test_elx_serialize.py @@ -1,9 +1,7 @@ -from unittest.mock import DEFAULT import itk itk.auto_progress(2) import pickle -import dask.distributed.protocol import itk_dreg.elastix.serialize diff --git a/src/test/itk_dreg/test_import.py b/src/test/itk_dreg/test_import.py index 0bdba66..a629825 100644 --- a/src/test/itk_dreg/test_import.py +++ b/src/test/itk_dreg/test_import.py @@ -12,11 +12,4 @@ def test_loadmodule(): - import itk_dreg - import itk_dreg.itk - import itk_dreg.base.image_block_interface - import itk_dreg.base.itk_typing - import itk_dreg.base.registration_interface - import itk_dreg.block.convert - import itk_dreg.block.image - import itk_dreg.register + pass diff --git a/src/test/itk_dreg/test_scheduler.py b/src/test/itk_dreg/test_scheduler.py index 381dbc4..7474dbb 100644 --- a/src/test/itk_dreg/test_scheduler.py +++ b/src/test/itk_dreg/test_scheduler.py @@ -8,7 +8,6 @@ import itk import numpy as np import dask -import dask.array as da import pytest from urllib.request import urlretrieve diff --git a/src/test/reduce_dfield/test_integration.py b/src/test/reduce_dfield/test_integration.py index dfbca15..fb7e3d7 100644 --- a/src/test/reduce_dfield/test_integration.py +++ b/src/test/reduce_dfield/test_integration.py @@ -8,7 +8,6 @@ import numpy as np import dask -import dask.array as da sys.path.append('./src') import itk_dreg.itk diff --git a/src/test/reduce_dfield/test_reduce_dfield.py b/src/test/reduce_dfield/test_reduce_dfield.py index b1ad2d6..7a2ddcb 100644 --- a/src/test/reduce_dfield/test_reduce_dfield.py +++ b/src/test/reduce_dfield/test_reduce_dfield.py @@ -13,11 +13,7 @@ def test_import(): - import itk_dreg.reduce_dfield - import itk_dreg.reduce_dfield.dreg - import itk_dreg.reduce_dfield.matrix_transform - import itk_dreg.reduce_dfield.transform_collection - import itk_dreg.reduce_dfield.transform + pass def test_collection_to_deformation_field_transform(): import itk_dreg.reduce_dfield.transform diff --git a/src/test/util/mock.py b/src/test/util/mock.py index 759bdb5..e04d591 100644 --- a/src/test/util/mock.py +++ b/src/test/util/mock.py @@ -1,10 +1,8 @@ -from unittest.mock import MagicMock -from typing import Optional, Iterable, Iterator +from typing import Iterable, Iterator import itk -from itk_dreg.base.itk_typing import ImageType, TransformType from itk_dreg.base.image_block_interface import BlockPairRegistrationResult, RegistrationTransformResult, BlockRegStatus, LocatedBlockResult from itk_dreg.base.registration_interface import BlockPairRegistrationMethod, ReduceResultsMethod From e7f60da209468cf3b1d628c7399b5ff139f8a397 Mon Sep 17 00:00:00 2001 From: Matt McCormick Date: Mon, 22 Apr 2024 14:33:28 -0400 Subject: [PATCH 4/7] STYLE: Address ruff lint checkers --- src/pyproject.toml | 2 +- src/test/elastix/test_elx_integration.py | 11 +++++---- src/test/elastix/test_elx_serialize.py | 3 ++- src/test/itk_dreg/test_block_interface.py | 24 +++++++++---------- src/test/itk_dreg/test_scheduler.py | 6 ++--- src/test/itk_dreg/test_serialize.py | 4 ++-- src/test/reduce_dfield/test_integration.py | 10 ++++---- .../test_transform_collection.py | 8 ++----- 8 files changed, 34 insertions(+), 34 deletions(-) diff --git a/src/pyproject.toml b/src/pyproject.toml index 80237ca..38da4f9 100644 --- a/src/pyproject.toml +++ b/src/pyproject.toml @@ -82,7 +82,7 @@ notebook = [ 'graphviz', ] -[tool.ruff] +[tool.ruff.lint] extend-ignore = ["E501"] [tool.hatch.version] diff --git a/src/test/elastix/test_elx_integration.py b/src/test/elastix/test_elx_integration.py index 588ec9f..b7eb9a2 100644 --- a/src/test/elastix/test_elx_integration.py +++ b/src/test/elastix/test_elx_integration.py @@ -8,7 +8,6 @@ import numpy as np import dask -sys.path.append('./src') import itk_dreg.itk from itk_dreg.register import register_images @@ -34,10 +33,10 @@ def test_run_dreg(): fixed_arr = np.ones([100]*3) moving_arr = np.random.random_sample([50]*3).astype(np.float32) - + register_method = ElastixDRegBlockPairRegistrationMethod() reduce_method = dreg_mock.CountingReduceResultsMethod() - + registration_result = None registration_schedule = None @@ -46,11 +45,13 @@ def test_run_dreg(): MOVING_FILEPATH = f'{testdir}/moving_image.mha' fixed_image = itk.image_view_from_array(fixed_arr) itk.imwrite(fixed_image, FIXED_FILEPATH, compression=False) - fixed_cb = lambda : itk_dreg.itk.make_reader(FIXED_FILEPATH) + def fixed_cb(): + return itk_dreg.itk.make_reader(FIXED_FILEPATH) moving_image = itk.image_view_from_array(moving_arr) moving_image.SetSpacing([2]*3) itk.imwrite(moving_image, MOVING_FILEPATH, compression=False) - moving_cb = lambda : itk_dreg.itk.make_reader(MOVING_FILEPATH) + def moving_cb(): + return itk_dreg.itk.make_reader(MOVING_FILEPATH) parameter_object = itk.ParameterObject.New() parameter_object.AddParameterMap( diff --git a/src/test/elastix/test_elx_serialize.py b/src/test/elastix/test_elx_serialize.py index a836642..93a080d 100644 --- a/src/test/elastix/test_elx_serialize.py +++ b/src/test/elastix/test_elx_serialize.py @@ -1,10 +1,11 @@ import itk -itk.auto_progress(2) import pickle import itk_dreg.elastix.serialize +itk.auto_progress(2) + def validate_parameter_maps(m1, m2): assert all([k in m2.keys() for k in m1.keys()]) assert all([k in m1.keys() for k in m2.keys()]) diff --git a/src/test/itk_dreg/test_block_interface.py b/src/test/itk_dreg/test_block_interface.py index cf1fa1f..33d4e65 100644 --- a/src/test/itk_dreg/test_block_interface.py +++ b/src/test/itk_dreg/test_block_interface.py @@ -1,20 +1,20 @@ - import numpy as np import itk -itk.auto_progress(2) import pytest from itk_dreg.base.image_block_interface import BlockRegStatus, BlockPairRegistrationResult +itk.auto_progress(2) + def test_construct_failed_pairwise_result(): # Verify no inputs required for failure result = BlockPairRegistrationResult(status=BlockRegStatus.FAILURE) assert result.status == BlockRegStatus.FAILURE - assert result.transform == None - assert result.transform_domain == None - assert result.inv_transform == None - assert result.inv_transform_domain == None + assert result.transform is None + assert result.transform_domain is None + assert result.inv_transform is None + assert result.inv_transform_domain is None def test_construct_forward_pairwise_result(): valid_transform = itk.TranslationTransform[itk.D,3].New() @@ -28,8 +28,8 @@ def test_construct_forward_pairwise_result(): assert result.status == BlockRegStatus.SUCCESS assert result.transform == valid_transform assert result.transform_domain == valid_transform_domain - assert result.inv_transform == None - assert result.inv_transform_domain == None + assert result.inv_transform is None + assert result.inv_transform_domain is None # Validate incomplete construction with pytest.raises(ValueError): @@ -38,7 +38,7 @@ def test_construct_forward_pairwise_result(): transform=valid_transform # transform_domain required ) - + invalid_transform_domain = [1,2,3] with pytest.raises(KeyError): BlockPairRegistrationResult( @@ -56,7 +56,7 @@ def test_construct_forward_pairwise_result(): transform_domain=invalid_transform_domain ) - + def test_construct_inverse_pairwise_result(): valid_transform = itk.TranslationTransform[itk.D,3].New() valid_transform_domain = itk.Image[itk.UC,3].New() @@ -79,7 +79,7 @@ def test_construct_inverse_pairwise_result(): assert result.transform_domain == valid_transform_domain assert result.inv_transform == valid_inverse_transform assert result.inv_transform_domain == valid_inverse_transform_domain - + # validate incomplete construction with pytest.raises(ValueError): BlockPairRegistrationResult( @@ -89,7 +89,7 @@ def test_construct_inverse_pairwise_result(): inv_transform=valid_transform, # inv_transform_domain required ) - + invalid_transform_domain = [1,2,3] with pytest.raises(KeyError): BlockPairRegistrationResult( diff --git a/src/test/itk_dreg/test_scheduler.py b/src/test/itk_dreg/test_scheduler.py index 7474dbb..3cba2bc 100644 --- a/src/test/itk_dreg/test_scheduler.py +++ b/src/test/itk_dreg/test_scheduler.py @@ -67,7 +67,7 @@ def moving_image_filepath(test_input_dir) -> str: image = itk.imread(f'{tmpdir}/HeadMRVolume_rigid.mha') itk.imwrite(image, MOVING_IMAGE_FILEPATH, compression=False) yield MOVING_IMAGE_FILEPATH - + def test_run_singlethreaded(fixed_image_filepath, moving_image_filepath, test_output_dir): dask.config.set(scheduler='single-threaded') @@ -109,7 +109,7 @@ def test_run_singlethreaded(fixed_image_filepath, moving_image_filepath, test_ou def test_localcluster(fixed_image_filepath, moving_image_filepath): import dask.distributed cluster = dask.distributed.LocalCluster(n_workers=1, threads_per_worker=1) - client = dask.distributed.Client(cluster) + client = dask.distributed.Client(cluster) # noqa: F841 # Methods import functools @@ -133,7 +133,7 @@ def test_localcluster(fixed_image_filepath, moving_image_filepath): reduce_method=reduce_method, overlap_factors=overlap_factors ) - + itk.auto_progress(0) registration_result = registration_graph.registration_result.compute() print(registration_result) diff --git a/src/test/itk_dreg/test_serialize.py b/src/test/itk_dreg/test_serialize.py index f56daff..6e31247 100644 --- a/src/test/itk_dreg/test_serialize.py +++ b/src/test/itk_dreg/test_serialize.py @@ -1,12 +1,12 @@ - import itk -itk.auto_progress(2) import pickle import dask.distributed.protocol import itk_dreg.base.image_block_interface +itk.auto_progress(2) + def test_serialize_pairwise_result(): failure_result = itk_dreg.base.image_block_interface.BlockPairRegistrationResult( status=itk_dreg.base.image_block_interface.BlockRegStatus.FAILURE diff --git a/src/test/reduce_dfield/test_integration.py b/src/test/reduce_dfield/test_integration.py index fb7e3d7..268bc2e 100644 --- a/src/test/reduce_dfield/test_integration.py +++ b/src/test/reduce_dfield/test_integration.py @@ -4,12 +4,10 @@ import tempfile import itk -itk.auto_progress(2) import numpy as np import dask -sys.path.append('./src') import itk_dreg.itk import itk_dreg.reduce_dfield.dreg from itk_dreg.register import register_images @@ -17,6 +15,8 @@ sys.path.append("./test") from util import mock as dreg_mock +itk.auto_progress(2) + """ Test the `itk_dreg` registration scheduling framework. """ @@ -38,11 +38,13 @@ def test_run_dreg(): MOVING_FILEPATH = f'{testdir}/moving_image.mha' fixed_image = itk.image_view_from_array(fixed_arr) itk.imwrite(fixed_image, FIXED_FILEPATH, compression=False) - fixed_cb = lambda : itk_dreg.itk.make_reader(FIXED_FILEPATH) + def fixed_cb(): + return itk_dreg.itk.make_reader(FIXED_FILEPATH) moving_image = itk.image_view_from_array(moving_arr) moving_image.SetSpacing([2]*3) itk.imwrite(moving_image, MOVING_FILEPATH, compression=False) - moving_cb = lambda : itk_dreg.itk.make_reader(MOVING_FILEPATH) + def moving_cb(): + return itk_dreg.itk.make_reader(MOVING_FILEPATH) registration_schedule = register_images( fixed_chunk_size=(10,20,100), diff --git a/src/test/reduce_dfield/test_transform_collection.py b/src/test/reduce_dfield/test_transform_collection.py index 6da4912..c4db541 100644 --- a/src/test/reduce_dfield/test_transform_collection.py +++ b/src/test/reduce_dfield/test_transform_collection.py @@ -1,17 +1,13 @@ #!/usr/bin/env python3 -import sys - -sys.path.append("src") - import itk import numpy as np import pytest -itk.auto_progress(2) - from itk_dreg.reduce_dfield.transform_collection import TransformEntry, TransformCollection +itk.auto_progress(2) + def test_unbounded_transform(): demo_transform = itk.TranslationTransform[itk.D,3].New() demo_transform.Translate([1,1,1]) From 67aa718558638812207f7afe8923e8c5e6acf233 Mon Sep 17 00:00:00 2001 From: Matt McCormick Date: Mon, 22 Apr 2024 14:34:53 -0400 Subject: [PATCH 5/7] STYLE: Increase black line length --- src/pyproject.toml | 3 +++ 1 file changed, 3 insertions(+) diff --git a/src/pyproject.toml b/src/pyproject.toml index 38da4f9..a268f58 100644 --- a/src/pyproject.toml +++ b/src/pyproject.toml @@ -101,3 +101,6 @@ dependencies = [ test = [ "pytest -vvv -s" ] + +[tool.black] +line-length = 88 \ No newline at end of file From 53774283bfa554609056e73c1a63f509062b5039 Mon Sep 17 00:00:00 2001 From: Matt McCormick Date: Mon, 22 Apr 2024 14:36:45 -0400 Subject: [PATCH 6/7] STYLE: run block on test sources --- src/test/elastix/test_elx_integration.py | 25 +-- src/test/elastix/test_elx_serialize.py | 25 ++- src/test/itk_dreg/test_block_interface.py | 72 ++++---- .../itk_dreg/test_dreg_block_subdivide.py | 123 ++++++++------ src/test/itk_dreg/test_scheduler.py | 88 ++++++---- src/test/itk_dreg/test_serialize.py | 9 +- src/test/reduce_dfield/test_integration.py | 31 ++-- src/test/reduce_dfield/test_reduce_dfield.py | 48 ++++-- .../test_transform_collection.py | 154 ++++++++++++------ src/test/util/mock.py | 65 ++++---- 10 files changed, 396 insertions(+), 244 deletions(-) diff --git a/src/test/elastix/test_elx_integration.py b/src/test/elastix/test_elx_integration.py index b7eb9a2..6617598 100644 --- a/src/test/elastix/test_elx_integration.py +++ b/src/test/elastix/test_elx_integration.py @@ -28,11 +28,12 @@ https://github.com/InsightSoftwareConsortium/ITKElastix/issues/255 """ + def test_run_dreg(): - dask.config.set(scheduler='single-threaded') + dask.config.set(scheduler="single-threaded") - fixed_arr = np.ones([100]*3) - moving_arr = np.random.random_sample([50]*3).astype(np.float32) + fixed_arr = np.ones([100] * 3) + moving_arr = np.random.random_sample([50] * 3).astype(np.float32) register_method = ElastixDRegBlockPairRegistrationMethod() reduce_method = dreg_mock.CountingReduceResultsMethod() @@ -41,30 +42,33 @@ def test_run_dreg(): registration_schedule = None with tempfile.TemporaryDirectory() as testdir: - FIXED_FILEPATH = f'{testdir}/fixed_image.mha' - MOVING_FILEPATH = f'{testdir}/moving_image.mha' + FIXED_FILEPATH = f"{testdir}/fixed_image.mha" + MOVING_FILEPATH = f"{testdir}/moving_image.mha" fixed_image = itk.image_view_from_array(fixed_arr) itk.imwrite(fixed_image, FIXED_FILEPATH, compression=False) + def fixed_cb(): return itk_dreg.itk.make_reader(FIXED_FILEPATH) + moving_image = itk.image_view_from_array(moving_arr) - moving_image.SetSpacing([2]*3) + moving_image.SetSpacing([2] * 3) itk.imwrite(moving_image, MOVING_FILEPATH, compression=False) + def moving_cb(): return itk_dreg.itk.make_reader(MOVING_FILEPATH) parameter_object = itk.ParameterObject.New() parameter_object.AddParameterMap( - parameter_object.GetDefaultParameterMap('rigid') + parameter_object.GetDefaultParameterMap("rigid") ) registration_schedule = register_images( - fixed_chunk_size=(10,20,100), - initial_transform=itk.TranslationTransform[itk.D,3].New(), + fixed_chunk_size=(10, 20, 100), + initial_transform=itk.TranslationTransform[itk.D, 3].New(), moving_reader_ctor=moving_cb, fixed_reader_ctor=fixed_cb, reduce_method=reduce_method, - overlap_factors=[0.1]*3, + overlap_factors=[0.1] * 3, block_registration_method=register_method, elx_parameter_object_serial=parameter_object_to_list(parameter_object), itk_transform_types=[itk.Euler3DTransform[itk.D]], @@ -76,4 +80,3 @@ def moving_cb(): assert reduce_method.num_calls == 1 assert registration_result.status.shape == registration_schedule.fixed_da.numblocks - diff --git a/src/test/elastix/test_elx_serialize.py b/src/test/elastix/test_elx_serialize.py index 93a080d..2e805ec 100644 --- a/src/test/elastix/test_elx_serialize.py +++ b/src/test/elastix/test_elx_serialize.py @@ -6,28 +6,37 @@ itk.auto_progress(2) + def validate_parameter_maps(m1, m2): assert all([k in m2.keys() for k in m1.keys()]) assert all([k in m1.keys() for k in m2.keys()]) for key in m1.keys(): assert m1[key] == m2[key] + def validate_parameter_objects(p1, p2): assert p1.GetNumberOfParameterMaps() == p2.GetNumberOfParameterMaps() for map_index in range(p1.GetNumberOfParameterMaps()): - validate_parameter_maps(p1.GetParameterMap(map_index), p2.GetParameterMap(map_index)) + validate_parameter_maps( + p1.GetParameterMap(map_index), p2.GetParameterMap(map_index) + ) + def test_serialize_elx_parameter_object(): - DEFAULT_PARAMETER_MAPS = ['rigid'] + DEFAULT_PARAMETER_MAPS = ["rigid"] parameter_object = itk.ParameterObject.New() parameter_object.AddParameterMap( itk.ParameterObject.GetDefaultParameterMap(DEFAULT_PARAMETER_MAPS[0]) ) - parameter_list = itk_dreg.elastix.serialize.parameter_object_to_list(parameter_object) + parameter_list = itk_dreg.elastix.serialize.parameter_object_to_list( + parameter_object + ) pickled_parameter_list = pickle.dumps(parameter_list) unpickled_parameter_list = pickle.loads(pickled_parameter_list) - unpickled_parameter_object = itk_dreg.elastix.serialize.list_to_parameter_object(unpickled_parameter_list) + unpickled_parameter_object = itk_dreg.elastix.serialize.list_to_parameter_object( + unpickled_parameter_list + ) validate_parameter_objects(unpickled_parameter_object, parameter_object) # TODO: https://github.com/InsightSoftwareConsortium/ITKElastix/issues/257 @@ -42,7 +51,7 @@ def test_serialize_elx_parameter_object(): # Validate baseline is unchanged assert parameter_object.GetNumberOfParameterMaps() == 1 - validate_parameter_maps(parameter_object.GetParameterMap(0), - itk.ParameterObject.GetDefaultParameterMap(DEFAULT_PARAMETER_MAPS[0])) - - + validate_parameter_maps( + parameter_object.GetParameterMap(0), + itk.ParameterObject.GetDefaultParameterMap(DEFAULT_PARAMETER_MAPS[0]), + ) diff --git a/src/test/itk_dreg/test_block_interface.py b/src/test/itk_dreg/test_block_interface.py index 33d4e65..ad7268b 100644 --- a/src/test/itk_dreg/test_block_interface.py +++ b/src/test/itk_dreg/test_block_interface.py @@ -3,10 +3,14 @@ import pytest -from itk_dreg.base.image_block_interface import BlockRegStatus, BlockPairRegistrationResult +from itk_dreg.base.image_block_interface import ( + BlockRegStatus, + BlockPairRegistrationResult, +) itk.auto_progress(2) + def test_construct_failed_pairwise_result(): # Verify no inputs required for failure result = BlockPairRegistrationResult(status=BlockRegStatus.FAILURE) @@ -16,14 +20,15 @@ def test_construct_failed_pairwise_result(): assert result.inv_transform is None assert result.inv_transform_domain is None + def test_construct_forward_pairwise_result(): - valid_transform = itk.TranslationTransform[itk.D,3].New() - valid_transform_domain = itk.Image[itk.UC,3].New() - valid_transform_domain.SetRegions([1,1,1]) + valid_transform = itk.TranslationTransform[itk.D, 3].New() + valid_transform_domain = itk.Image[itk.UC, 3].New() + valid_transform_domain.SetRegions([1, 1, 1]) result = BlockPairRegistrationResult( status=BlockRegStatus.SUCCESS, transform=valid_transform, - transform_domain=valid_transform_domain + transform_domain=valid_transform_domain, ) assert result.status == BlockRegStatus.SUCCESS assert result.transform == valid_transform @@ -35,44 +40,51 @@ def test_construct_forward_pairwise_result(): with pytest.raises(ValueError): BlockPairRegistrationResult( status=BlockRegStatus.SUCCESS, - transform=valid_transform + transform=valid_transform, # transform_domain required ) - invalid_transform_domain = [1,2,3] + invalid_transform_domain = [1, 2, 3] with pytest.raises(KeyError): BlockPairRegistrationResult( status=BlockRegStatus.SUCCESS, transform=valid_transform, - transform_domain=invalid_transform_domain + transform_domain=invalid_transform_domain, ) - invalid_transform_domain = itk.Image[itk.UC,3].New() - assert all([size == 0 for size in invalid_transform_domain.GetLargestPossibleRegion().GetSize()]) + invalid_transform_domain = itk.Image[itk.UC, 3].New() + assert all( + [ + size == 0 + for size in invalid_transform_domain.GetLargestPossibleRegion().GetSize() + ] + ) with pytest.raises(ValueError): BlockPairRegistrationResult( status=BlockRegStatus.SUCCESS, transform=valid_transform, - transform_domain=invalid_transform_domain + transform_domain=invalid_transform_domain, ) def test_construct_inverse_pairwise_result(): - valid_transform = itk.TranslationTransform[itk.D,3].New() - valid_transform_domain = itk.Image[itk.UC,3].New() - valid_transform_domain.SetRegions([1,1,1]) - valid_inverse_transform = itk.TranslationTransform[itk.D,3].New() - valid_inverse_transform_domain = itk.Image[itk.UC,3].New() - valid_inverse_transform_domain.SetRegions([1,1,1]) - valid_inverse_transform_domain.SetOrigin([-1]*3) - valid_inverse_transform_domain.SetSpacing([0.1]*3) - valid_inverse_transform_domain.SetDirection(np.array([[0,-1,0],[-1,0,0],[0,0,1]])) + valid_transform = itk.TranslationTransform[itk.D, 3].New() + valid_transform_domain = itk.Image[itk.UC, 3].New() + valid_transform_domain.SetRegions([1, 1, 1]) + valid_inverse_transform = itk.TranslationTransform[itk.D, 3].New() + valid_inverse_transform_domain = itk.Image[itk.UC, 3].New() + valid_inverse_transform_domain.SetRegions([1, 1, 1]) + valid_inverse_transform_domain.SetOrigin([-1] * 3) + valid_inverse_transform_domain.SetSpacing([0.1] * 3) + valid_inverse_transform_domain.SetDirection( + np.array([[0, -1, 0], [-1, 0, 0], [0, 0, 1]]) + ) result = BlockPairRegistrationResult( status=BlockRegStatus.SUCCESS, transform=valid_transform, transform_domain=valid_transform_domain, inv_transform=valid_inverse_transform, - inv_transform_domain=valid_inverse_transform_domain + inv_transform_domain=valid_inverse_transform_domain, ) assert result.status == BlockRegStatus.SUCCESS assert result.transform == valid_transform @@ -90,26 +102,28 @@ def test_construct_inverse_pairwise_result(): # inv_transform_domain required ) - invalid_transform_domain = [1,2,3] + invalid_transform_domain = [1, 2, 3] with pytest.raises(KeyError): BlockPairRegistrationResult( status=BlockRegStatus.SUCCESS, transform=valid_transform, transform_domain=valid_transform_domain, inv_transform=valid_transform, - inv_transform_domain=invalid_transform_domain + inv_transform_domain=invalid_transform_domain, ) - invalid_transform_domain = itk.Image[itk.UC,3].New() - assert all([size == 0 for size in invalid_transform_domain.GetLargestPossibleRegion().GetSize()]) + invalid_transform_domain = itk.Image[itk.UC, 3].New() + assert all( + [ + size == 0 + for size in invalid_transform_domain.GetLargestPossibleRegion().GetSize() + ] + ) with pytest.raises(ValueError): BlockPairRegistrationResult( status=BlockRegStatus.SUCCESS, transform=valid_transform, transform_domain=valid_transform_domain, inv_transform=valid_transform, - inv_transform_domain=invalid_transform_domain + inv_transform_domain=invalid_transform_domain, ) - - - diff --git a/src/test/itk_dreg/test_dreg_block_subdivide.py b/src/test/itk_dreg/test_dreg_block_subdivide.py index a36b4b2..0c2a57c 100644 --- a/src/test/itk_dreg/test_dreg_block_subdivide.py +++ b/src/test/itk_dreg/test_dreg_block_subdivide.py @@ -20,33 +20,39 @@ before any sampling/resampling is performed. """ + def test_rescale_physical_region_norescale(): SCALE_FACTORS = [1] * 3 IMAGE_SIZE = [10] * 3 - input_image = itk.Image[itk.F,3].New() + input_image = itk.Image[itk.F, 3].New() input_image.SetRegions(IMAGE_SIZE) physical_region = itk_dreg.block.convert.image_to_physical_region( - image_region=input_image.GetLargestPossibleRegion(), - ref_image=input_image - ) + image_region=input_image.GetLargestPossibleRegion(), ref_image=input_image + ) output_image = itk_dreg.block.image.physical_region_to_itk_image( physical_region=physical_region, - spacing = [spacing * scale for spacing, scale in zip(itk.spacing(input_image), SCALE_FACTORS)], + spacing=[ + spacing * scale + for spacing, scale in zip(itk.spacing(input_image), SCALE_FACTORS) + ], direction=np.array(input_image.GetDirection()), - extend_beyond=False + extend_beyond=False, ) assert itk.size(output_image) == itk.size(input_image) assert itk.origin(output_image) == itk.origin(input_image) assert itk.spacing(output_image) == itk.spacing(input_image) - assert np.all(itk_dreg.block.image.get_sample_bounds(output_image) == - itk_dreg.block.image.get_sample_bounds(input_image)) - + assert np.all( + itk_dreg.block.image.get_sample_bounds(output_image) + == itk_dreg.block.image.get_sample_bounds(input_image) + ) + + def test_rescale_physical_region_downscale(): SCALE_FACTORS = [2] * 3 IMAGE_SIZE = [10] * 3 - input_image = itk.Image[itk.F,3].New() + input_image = itk.Image[itk.F, 3].New() input_image.SetRegions(IMAGE_SIZE) EXPECTED_SIZE = [5] * 3 @@ -54,25 +60,28 @@ def test_rescale_physical_region_downscale(): EXPECTED_ORIGIN = [0.5] * 3 physical_region = itk_dreg.block.convert.image_to_physical_region( - image_region=input_image.GetLargestPossibleRegion(), - ref_image=input_image - ) + image_region=input_image.GetLargestPossibleRegion(), ref_image=input_image + ) output_image = itk_dreg.block.image.physical_region_to_itk_image( physical_region=physical_region, - spacing = [spacing * scale for spacing, scale in zip(itk.spacing(input_image), SCALE_FACTORS)], + spacing=[ + spacing * scale + for spacing, scale in zip(itk.spacing(input_image), SCALE_FACTORS) + ], direction=np.array(input_image.GetDirection()), - extend_beyond=False + extend_beyond=False, ) assert itk.size(output_image) == EXPECTED_SIZE assert itk.spacing(output_image) == EXPECTED_SPACING assert itk.origin(output_image) == EXPECTED_ORIGIN + def test_rescale_physical_region_offset(): SCALE_FACTORS = [2] * 3 IMAGE_SIZE = [10] * 3 IMAGE_INDEX = [10] * 3 - input_image = itk.Image[itk.F,3].New() + input_image = itk.Image[itk.F, 3].New() image_region = itk.ImageRegion[3]() image_region.SetIndex(IMAGE_INDEX) @@ -85,14 +94,16 @@ def test_rescale_physical_region_offset(): EXPECTED_SIZE = [5] * 3 physical_region = itk_dreg.block.convert.image_to_physical_region( - image_region=input_image.GetLargestPossibleRegion(), - ref_image=input_image - ) + image_region=input_image.GetLargestPossibleRegion(), ref_image=input_image + ) output_image = itk_dreg.block.image.physical_region_to_itk_image( physical_region=physical_region, - spacing = [spacing * scale for spacing, scale in zip(itk.spacing(input_image), SCALE_FACTORS)], + spacing=[ + spacing * scale + for spacing, scale in zip(itk.spacing(input_image), SCALE_FACTORS) + ], direction=np.array(input_image.GetDirection()), - extend_beyond=False + extend_beyond=False, ) assert itk.size(output_image) == EXPECTED_SIZE @@ -103,11 +114,11 @@ def test_rescale_physical_region_offset(): def test_rescale_physical_region_requested(): SCALE_FACTORS = [2] * 3 - IMAGE_SIZE=[100]*3 - input_image = itk.Image[itk.F,3].New() + IMAGE_SIZE = [100] * 3 + input_image = itk.Image[itk.F, 3].New() input_image.SetRegions(IMAGE_SIZE) - REQUESTED_REGION = itk.ImageRegion[3]([1]*3,[10]*3) + REQUESTED_REGION = itk.ImageRegion[3]([1] * 3, [10] * 3) assert input_image.GetLargestPossibleRegion().IsInside(REQUESTED_REGION) EXPECTED_ORIGIN = [1.5] * 3 @@ -115,42 +126,46 @@ def test_rescale_physical_region_requested(): EXPECTED_SIZE = [5] * 3 physical_region = itk_dreg.block.convert.image_to_physical_region( - image_region=REQUESTED_REGION, - ref_image=input_image - ) + image_region=REQUESTED_REGION, ref_image=input_image + ) output_image = itk_dreg.block.image.physical_region_to_itk_image( physical_region=physical_region, - spacing = [spacing * scale for spacing, scale in zip(itk.spacing(input_image), SCALE_FACTORS)], + spacing=[ + spacing * scale + for spacing, scale in zip(itk.spacing(input_image), SCALE_FACTORS) + ], direction=np.array(input_image.GetDirection()), - extend_beyond=False + extend_beyond=False, ) assert itk.size(output_image) == EXPECTED_SIZE assert itk.spacing(output_image) == EXPECTED_SPACING assert itk.origin(output_image) == EXPECTED_ORIGIN - assert np.all(itk_dreg.block.image.get_sample_bounds(output_image) == - itk_dreg.block.convert.image_to_physical_region( - REQUESTED_REGION, - ref_image=input_image - )) - + assert np.all( + itk_dreg.block.image.get_sample_bounds(output_image) + == itk_dreg.block.convert.image_to_physical_region( + REQUESTED_REGION, ref_image=input_image + ) + ) + def test_rescale_physical_region_with_direction(): SCALE_FACTORS = [2] * 3 - IMAGE_SIZE=[100]*3 - IMAGE_DIRECTION = np.array([[0,0,-1],[1,0,0],[0,-1,0]]) - input_image = itk.Image[itk.F,3].New() + IMAGE_SIZE = [100] * 3 + IMAGE_DIRECTION = np.array([[0, 0, -1], [1, 0, 0], [0, -1, 0]]) + input_image = itk.Image[itk.F, 3].New() input_image.SetRegions(IMAGE_SIZE) input_image.SetDirection(IMAGE_DIRECTION) - assert input_image.TransformIndexToPhysicalPoint([25]*3) == [-25,25,-25] + assert input_image.TransformIndexToPhysicalPoint([25] * 3) == [-25, 25, -25] - REQUESTED_REGION = itk.ImageRegion[3]([25]*3,[15]*3) + REQUESTED_REGION = itk.ImageRegion[3]([25] * 3, [15] * 3) assert input_image.GetLargestPossibleRegion().IsInside(REQUESTED_REGION) - assert np.all(itk_dreg.block.convert.image_to_physical_region(REQUESTED_REGION, input_image) ==\ - np.array([[-39.5, 24.5, -39.5], - [-24.5, 39.5, -24.5]])) + assert np.all( + itk_dreg.block.convert.image_to_physical_region(REQUESTED_REGION, input_image) + == np.array([[-39.5, 24.5, -39.5], [-24.5, 39.5, -24.5]]) + ) # The input physical region cannot be evenly subdivided into a voxel grid with the given spacing, # so allow the grid to extend beyond the input physical region by up to 1 voxel width. @@ -159,20 +174,19 @@ def test_rescale_physical_region_with_direction(): EXPECTED_ORIGIN = [-25, 25, -25] EXPECTED_SPACING = SCALE_FACTORS EXPECTED_SIZE = [8] * 3 - EXPECTED_PHYSICAL_REGION = np.array([ - [-40, 24, -40], - [-24, 40, -24] - ]) + EXPECTED_PHYSICAL_REGION = np.array([[-40, 24, -40], [-24, 40, -24]]) physical_region = itk_dreg.block.convert.image_to_physical_region( - image_region=REQUESTED_REGION, - ref_image=input_image - ) + image_region=REQUESTED_REGION, ref_image=input_image + ) output_image = itk_dreg.block.image.physical_region_to_itk_image( physical_region=physical_region, - spacing = [spacing * scale for spacing, scale in zip(itk.spacing(input_image), SCALE_FACTORS)], + spacing=[ + spacing * scale + for spacing, scale in zip(itk.spacing(input_image), SCALE_FACTORS) + ], direction=IMAGE_DIRECTION, - extend_beyond=EXTEND_BEYOND + extend_beyond=EXTEND_BEYOND, ) assert itk.size(output_image) == EXPECTED_SIZE @@ -180,5 +194,6 @@ def test_rescale_physical_region_with_direction(): assert itk.origin(output_image) == EXPECTED_ORIGIN assert output_image.GetDirection() == input_image.GetDirection() - assert np.all(itk_dreg.block.image.get_sample_bounds(output_image) == - EXPECTED_PHYSICAL_REGION) + assert np.all( + itk_dreg.block.image.get_sample_bounds(output_image) == EXPECTED_PHYSICAL_REGION + ) diff --git a/src/test/itk_dreg/test_scheduler.py b/src/test/itk_dreg/test_scheduler.py index 3cba2bc..55ee14b 100644 --- a/src/test/itk_dreg/test_scheduler.py +++ b/src/test/itk_dreg/test_scheduler.py @@ -30,62 +30,79 @@ # worker_logger.setLevel(logging.DEBUG) PIXEL_TYPE = itk.F -DIMENSION = 3 # 2D is planned but not yet supported (2023.10.20) +DIMENSION = 3 # 2D is planned but not yet supported (2023.10.20) + @pytest.fixture def test_input_dir() -> str: - TEST_INPUT_DIR = 'test/data/input' - os.makedirs(TEST_INPUT_DIR,exist_ok=True) + TEST_INPUT_DIR = "test/data/input" + os.makedirs(TEST_INPUT_DIR, exist_ok=True) yield TEST_INPUT_DIR + @pytest.fixture def test_output_dir() -> str: - TEST_OUTPUT_DIR = 'test/data/output/itk_dreg' + TEST_OUTPUT_DIR = "test/data/output/itk_dreg" os.makedirs(TEST_OUTPUT_DIR, exist_ok=True) return TEST_OUTPUT_DIR + @pytest.fixture def fixed_image_filepath(test_input_dir): # A small 3D sample MRI image depicting a patient's head - FIXED_IMAGE_FILEPATH = os.path.abspath(f'{test_input_dir}/HeadMRVolume_d.mha') - COMPRESSED_FIXED_IMAGE_URL = 'https://data.kitware.com/api/v1/item/65328e0a5be10c8fb6ed4f01/download' + FIXED_IMAGE_FILEPATH = os.path.abspath(f"{test_input_dir}/HeadMRVolume_d.mha") + COMPRESSED_FIXED_IMAGE_URL = ( + "https://data.kitware.com/api/v1/item/65328e0a5be10c8fb6ed4f01/download" + ) if not os.path.exists(FIXED_IMAGE_FILEPATH): with tempfile.TemporaryDirectory(dir=test_input_dir) as tmpdir: - urlretrieve(COMPRESSED_FIXED_IMAGE_URL, f'{tmpdir}/HeadMRVolume.mha') - image = itk.imread(f'{tmpdir}/HeadMRVolume.mha') + urlretrieve(COMPRESSED_FIXED_IMAGE_URL, f"{tmpdir}/HeadMRVolume.mha") + image = itk.imread(f"{tmpdir}/HeadMRVolume.mha") itk.imwrite(image, FIXED_IMAGE_FILEPATH, compression=False) yield FIXED_IMAGE_FILEPATH + @pytest.fixture def moving_image_filepath(test_input_dir) -> str: # The fixed image, but with an arbitrary translation and rotation applied - MOVING_IMAGE_FILEPATH = os.path.abspath(f'{test_input_dir}/HeadMRVolume_rigid_d.mha') - COMPRESSED_MOVING_IMAGE_URL = 'https://data.kitware.com/api/v1/item/65328e0f5be10c8fb6ed4f04/download' + MOVING_IMAGE_FILEPATH = os.path.abspath( + f"{test_input_dir}/HeadMRVolume_rigid_d.mha" + ) + COMPRESSED_MOVING_IMAGE_URL = ( + "https://data.kitware.com/api/v1/item/65328e0f5be10c8fb6ed4f04/download" + ) if not os.path.exists(MOVING_IMAGE_FILEPATH): with tempfile.TemporaryDirectory(dir=test_input_dir) as tmpdir: - urlretrieve(COMPRESSED_MOVING_IMAGE_URL, f'{tmpdir}/HeadMRVolume_rigid.mha') - image = itk.imread(f'{tmpdir}/HeadMRVolume_rigid.mha') + urlretrieve(COMPRESSED_MOVING_IMAGE_URL, f"{tmpdir}/HeadMRVolume_rigid.mha") + image = itk.imread(f"{tmpdir}/HeadMRVolume_rigid.mha") itk.imwrite(image, MOVING_IMAGE_FILEPATH, compression=False) yield MOVING_IMAGE_FILEPATH -def test_run_singlethreaded(fixed_image_filepath, moving_image_filepath, test_output_dir): - dask.config.set(scheduler='single-threaded') +def test_run_singlethreaded( + fixed_image_filepath, moving_image_filepath, test_output_dir +): + dask.config.set(scheduler="single-threaded") logging.root.setLevel(logging.INFO) # Methods import functools - fixed_reader_ctor = functools.partial(itk_dreg.itk.make_reader, filepath=fixed_image_filepath) - moving_reader_ctor = functools.partial(itk_dreg.itk.make_reader, filepath=moving_image_filepath) + + fixed_reader_ctor = functools.partial( + itk_dreg.itk.make_reader, filepath=fixed_image_filepath + ) + moving_reader_ctor = functools.partial( + itk_dreg.itk.make_reader, filepath=moving_image_filepath + ) register_method = dreg_mock.CountingBlockPairRegistrationMethod() - logger.warning(f'reg def {register_method.default_result}') + logger.warning(f"reg def {register_method.default_result}") reduce_method = dreg_mock.CountingReduceResultsMethod() - logger.warning(f'reduce def {reduce_method.default_result}') + logger.warning(f"reduce def {reduce_method.default_result}") # Data - fixed_chunk_size=(10,100,100) #TODO investigate failure case (15,15,25) - initial_transform = itk.TranslationTransform[itk.D,3].New() - overlap_factors=[0.1]*3 + fixed_chunk_size = (10, 100, 100) # TODO investigate failure case (15,15,25) + initial_transform = itk.TranslationTransform[itk.D, 3].New() + overlap_factors = [0.1] * 3 registration_graph = itk_dreg.register.register_images( fixed_chunk_size=fixed_chunk_size, @@ -94,36 +111,44 @@ def test_run_singlethreaded(fixed_image_filepath, moving_image_filepath, test_ou fixed_reader_ctor=fixed_reader_ctor, block_registration_method=register_method, reduce_method=reduce_method, - overlap_factors=overlap_factors + overlap_factors=overlap_factors, ) itk.auto_progress(0) registration_result = registration_graph.registration_result.compute() print(registration_result) - assert register_method.num_calls == np.product(registration_graph.fixed_da.numblocks) + assert register_method.num_calls == np.product( + registration_graph.fixed_da.numblocks + ) assert reduce_method.num_calls == 1 assert registration_result.status.shape == registration_graph.fixed_da.numblocks def test_localcluster(fixed_image_filepath, moving_image_filepath): import dask.distributed + cluster = dask.distributed.LocalCluster(n_workers=1, threads_per_worker=1) - client = dask.distributed.Client(cluster) # noqa: F841 + client = dask.distributed.Client(cluster) # noqa: F841 # Methods import functools - fixed_reader_ctor = functools.partial(itk_dreg.itk.make_reader, filepath=fixed_image_filepath) - moving_reader_ctor = functools.partial(itk_dreg.itk.make_reader, filepath=moving_image_filepath) + + fixed_reader_ctor = functools.partial( + itk_dreg.itk.make_reader, filepath=fixed_image_filepath + ) + moving_reader_ctor = functools.partial( + itk_dreg.itk.make_reader, filepath=moving_image_filepath + ) register_method = dreg_mock.ConstantBlockPairRegistrationMethod() reduce_method = dreg_mock.ConstantReduceResultsMethod() # Data - fixed_chunk_size=(10,100,100) #TODO investigate failure case (15,15,25) - initial_transform = itk.TranslationTransform[itk.D,3].New() - overlap_factors=[0.1]*3 + fixed_chunk_size = (10, 100, 100) # TODO investigate failure case (15,15,25) + initial_transform = itk.TranslationTransform[itk.D, 3].New() + overlap_factors = [0.1] * 3 - #logging.root.setLevel(logging.DEBUG) + # logging.root.setLevel(logging.DEBUG) registration_graph = itk_dreg.register.register_images( fixed_chunk_size=fixed_chunk_size, initial_transform=initial_transform, @@ -131,7 +156,7 @@ def test_localcluster(fixed_image_filepath, moving_image_filepath): fixed_reader_ctor=fixed_reader_ctor, block_registration_method=register_method, reduce_method=reduce_method, - overlap_factors=overlap_factors + overlap_factors=overlap_factors, ) itk.auto_progress(0) @@ -140,4 +165,3 @@ def test_localcluster(fixed_image_filepath, moving_image_filepath): assert registration_result.status.shape == registration_graph.fixed_da.numblocks assert np.all(registration_result.status == BlockRegStatus.SUCCESS) - diff --git a/src/test/itk_dreg/test_serialize.py b/src/test/itk_dreg/test_serialize.py index 6e31247..e876b61 100644 --- a/src/test/itk_dreg/test_serialize.py +++ b/src/test/itk_dreg/test_serialize.py @@ -7,6 +7,7 @@ itk.auto_progress(2) + def test_serialize_pairwise_result(): failure_result = itk_dreg.base.image_block_interface.BlockPairRegistrationResult( status=itk_dreg.base.image_block_interface.BlockRegStatus.FAILURE @@ -18,12 +19,12 @@ def test_serialize_pairwise_result(): ) assert deserialized_result.status == failure_result.status - transform_domain = itk.Image[itk.F,3].New() - transform_domain.SetRegions([10]*3) + transform_domain = itk.Image[itk.F, 3].New() + transform_domain.SetRegions([10] * 3) success_result = itk_dreg.base.image_block_interface.BlockPairRegistrationResult( status=itk_dreg.base.image_block_interface.BlockRegStatus.SUCCESS, - transform=itk.TranslationTransform[itk.D,3].New(), - transform_domain=transform_domain + transform=itk.TranslationTransform[itk.D, 3].New(), + transform_domain=transform_domain, ) # TODO Unbuffered `itk.Image` is not yet pickleable (ITK v5.4rc2) # ValueError: PyMemoryView_FromBuffer(): info->buf must not be NULL diff --git a/src/test/reduce_dfield/test_integration.py b/src/test/reduce_dfield/test_integration.py index 268bc2e..94771b1 100644 --- a/src/test/reduce_dfield/test_integration.py +++ b/src/test/reduce_dfield/test_integration.py @@ -21,12 +21,13 @@ Test the `itk_dreg` registration scheduling framework. """ + def test_run_dreg(): - dask.config.set(scheduler='single-threaded') + dask.config.set(scheduler="single-threaded") + + fixed_arr = np.ones([100] * 3) + moving_arr = np.ones([50] * 3) - fixed_arr = np.ones([100]*3) - moving_arr = np.ones([50]*3) - register_method = dreg_mock.CountingBlockPairRegistrationMethod() reduce_method = itk_dreg.reduce_dfield.dreg.ReduceToDisplacementFieldMethod() @@ -34,33 +35,37 @@ def test_run_dreg(): registration_schedule = None with tempfile.TemporaryDirectory() as testdir: - FIXED_FILEPATH = f'{testdir}/fixed_image.mha' - MOVING_FILEPATH = f'{testdir}/moving_image.mha' + FIXED_FILEPATH = f"{testdir}/fixed_image.mha" + MOVING_FILEPATH = f"{testdir}/moving_image.mha" fixed_image = itk.image_view_from_array(fixed_arr) itk.imwrite(fixed_image, FIXED_FILEPATH, compression=False) + def fixed_cb(): return itk_dreg.itk.make_reader(FIXED_FILEPATH) + moving_image = itk.image_view_from_array(moving_arr) - moving_image.SetSpacing([2]*3) + moving_image.SetSpacing([2] * 3) itk.imwrite(moving_image, MOVING_FILEPATH, compression=False) + def moving_cb(): return itk_dreg.itk.make_reader(MOVING_FILEPATH) registration_schedule = register_images( - fixed_chunk_size=(10,20,100), - initial_transform=itk.TranslationTransform[itk.D,3].New(), + fixed_chunk_size=(10, 20, 100), + initial_transform=itk.TranslationTransform[itk.D, 3].New(), moving_reader_ctor=moving_cb, fixed_reader_ctor=fixed_cb, block_registration_method=register_method, reduce_method=reduce_method, - overlap_factors=[0.1]*3, - displacement_grid_scale_factors=[10.0,10.0,10.0] + overlap_factors=[0.1] * 3, + displacement_grid_scale_factors=[10.0, 10.0, 10.0], ) registration_result = registration_schedule.registration_result.compute() print(registration_result) - assert register_method.num_calls == np.product(registration_schedule.fixed_da.numblocks) + assert register_method.num_calls == np.product( + registration_schedule.fixed_da.numblocks + ) assert registration_result.status.shape == registration_schedule.fixed_da.numblocks - diff --git a/src/test/reduce_dfield/test_reduce_dfield.py b/src/test/reduce_dfield/test_reduce_dfield.py index 7a2ddcb..7a60fc4 100644 --- a/src/test/reduce_dfield/test_reduce_dfield.py +++ b/src/test/reduce_dfield/test_reduce_dfield.py @@ -15,33 +15,49 @@ def test_import(): pass + def test_collection_to_deformation_field_transform(): import itk_dreg.reduce_dfield.transform import itk_dreg.reduce_dfield.transform_collection INPUT_SIZE = [10] * 3 SCALE_FACTORS = [2] * 3 - EXPECTED_OUTPUT_SIZE = [size / scale for size, scale in zip(INPUT_SIZE, SCALE_FACTORS)] + EXPECTED_OUTPUT_SIZE = [ + size / scale for size, scale in zip(INPUT_SIZE, SCALE_FACTORS) + ] TRANSLATION_COMPONENT = 1 - input_transform = itk.TranslationTransform[itk.D,3].New() + input_transform = itk.TranslationTransform[itk.D, 3].New() input_transform.Translate([TRANSLATION_COMPONENT] * 3) - reference_image = itk.Image[itk.F,3].New() - reference_image.SetRegions([10,10,10]) + reference_image = itk.Image[itk.F, 3].New() + reference_image.SetRegions([10, 10, 10]) transforms = itk_dreg.reduce_dfield.transform_collection.TransformCollection( - transform_and_domain_list=[itk_dreg.reduce_dfield.transform_collection.TransformEntry(input_transform, None)] + transform_and_domain_list=[ + itk_dreg.reduce_dfield.transform_collection.TransformEntry( + input_transform, None + ) + ] ) - output_transform = itk_dreg.reduce_dfield.transform.collection_to_deformation_field_transform(transforms, - reference_image=reference_image, - initial_transform=itk.TranslationTransform[itk.D,3].New(), - scale_factors=SCALE_FACTORS) - - assert all([output_size == expected_size - for output_size, expected_size - in zip(itk.size(output_transform.GetDisplacementField()), EXPECTED_OUTPUT_SIZE)]),\ - f'Output has size {itk.size(output_transform.GetDisplacementField())}' - assert np.all(itk.array_view_from_image(output_transform.GetDisplacementField()) == TRANSLATION_COMPONENT) - + output_transform = ( + itk_dreg.reduce_dfield.transform.collection_to_deformation_field_transform( + transforms, + reference_image=reference_image, + initial_transform=itk.TranslationTransform[itk.D, 3].New(), + scale_factors=SCALE_FACTORS, + ) + ) + assert all( + [ + output_size == expected_size + for output_size, expected_size in zip( + itk.size(output_transform.GetDisplacementField()), EXPECTED_OUTPUT_SIZE + ) + ] + ), f"Output has size {itk.size(output_transform.GetDisplacementField())}" + assert np.all( + itk.array_view_from_image(output_transform.GetDisplacementField()) + == TRANSLATION_COMPONENT + ) diff --git a/src/test/reduce_dfield/test_transform_collection.py b/src/test/reduce_dfield/test_transform_collection.py index c4db541..dde1375 100644 --- a/src/test/reduce_dfield/test_transform_collection.py +++ b/src/test/reduce_dfield/test_transform_collection.py @@ -4,43 +4,57 @@ import numpy as np import pytest -from itk_dreg.reduce_dfield.transform_collection import TransformEntry, TransformCollection +from itk_dreg.reduce_dfield.transform_collection import ( + TransformEntry, + TransformCollection, +) itk.auto_progress(2) + def test_unbounded_transform(): - demo_transform = itk.TranslationTransform[itk.D,3].New() - demo_transform.Translate([1,1,1]) + demo_transform = itk.TranslationTransform[itk.D, 3].New() + demo_transform.Translate([1, 1, 1]) transforms = [TransformEntry(demo_transform, None)] transform_collection = TransformCollection( blend_method=TransformCollection.blend_simple_mean, - transform_and_domain_list=transforms + transform_and_domain_list=transforms, ) assert len(transform_collection.transforms) == 1 assert len(transform_collection.domains) == 1 assert transform_collection.domains[0] is None - assert np.all(transform_collection.transform_point(itk.Point[itk.F,3]([0,0,0])) == [1,1,1]) - assert np.all(transform_collection.transform_point(itk.Point[itk.F,3]([5,10,15])) == [6,11,16]) - assert np.all(transform_collection.transform_point(itk.Point[itk.F,3]([-30,-0.1,-0.2])) == [-29,0.9,0.8]) + assert np.all( + transform_collection.transform_point(itk.Point[itk.F, 3]([0, 0, 0])) + == [1, 1, 1] + ) + assert np.all( + transform_collection.transform_point(itk.Point[itk.F, 3]([5, 10, 15])) + == [6, 11, 16] + ) + assert np.all( + transform_collection.transform_point(itk.Point[itk.F, 3]([-30, -0.1, -0.2])) + == [-29, 0.9, 0.8] + ) + def test_bounded_transform(): - demo_transform = itk.TranslationTransform[itk.D,3].New() - demo_transform.Translate([1,1,1]) - demo_domain = itk.Image[itk.F,3].New() - demo_domain.SetOrigin([1,1,1]) + demo_transform = itk.TranslationTransform[itk.D, 3].New() + demo_transform.Translate([1, 1, 1]) + demo_domain = itk.Image[itk.F, 3].New() + demo_domain.SetOrigin([1, 1, 1]) r = itk.ImageRegion[3]() - r.SetSize([1,1,1]) + r.SetSize([1, 1, 1]) demo_domain.SetLargestPossibleRegion(r) # no allocate -- use itk.Image as metadata container transforms = [TransformEntry(demo_transform, demo_domain)] transform_collection = TransformCollection( blend_method=TransformCollection.blend_simple_mean, - transform_and_domain_list=transforms + transform_and_domain_list=transforms, ) assert len(transform_collection.transforms) == 1 @@ -48,63 +62,109 @@ def test_bounded_transform(): assert transform_collection.domains[0] == demo_domain with pytest.raises(Exception): - print(transform_collection.transform_point([0,0,0])) + print(transform_collection.transform_point([0, 0, 0])) + + assert np.all( + transform_collection.transform_point(itk.Point[itk.F, 3]([1, 1, 1])) + == [2, 2, 2] + ) - assert np.all(transform_collection.transform_point(itk.Point[itk.F,3]([1,1,1])) == [2,2,2]) def test_two_bounded_transforms(): - transforms = [TransformEntry(itk.TranslationTransform[itk.D,3].New(), itk.Image[itk.F,3].New()), - TransformEntry(itk.TranslationTransform[itk.D,3].New(), itk.Image[itk.F,3].New())] - transforms[0].transform.Translate([1,1,1]) - transforms[1].transform.Translate([2,2,2]) - transforms[0].domain.SetOrigin([0,0,0]) - transforms[0].domain.SetRegions([2,2,2]) - transforms[1].domain.SetOrigin([1,1,1]) - transforms[1].domain.SetRegions([2,2,2]) + transforms = [ + TransformEntry( + itk.TranslationTransform[itk.D, 3].New(), itk.Image[itk.F, 3].New() + ), + TransformEntry( + itk.TranslationTransform[itk.D, 3].New(), itk.Image[itk.F, 3].New() + ), + ] + transforms[0].transform.Translate([1, 1, 1]) + transforms[1].transform.Translate([2, 2, 2]) + transforms[0].domain.SetOrigin([0, 0, 0]) + transforms[0].domain.SetRegions([2, 2, 2]) + transforms[1].domain.SetOrigin([1, 1, 1]) + transforms[1].domain.SetRegions([2, 2, 2]) transform_collection = TransformCollection( blend_method=TransformCollection.blend_simple_mean, - transform_and_domain_list=transforms) + transform_and_domain_list=transforms, + ) assert len(transform_collection.transforms) == 2 assert len(transform_collection.domains) == 2 # Transform over non-overlapping domain region - assert np.all(transform_collection.transform_point(itk.Point[itk.F,3]([0,0,0])) == [1,1,1]) - assert np.all(transform_collection.transform_point(itk.Point[itk.F,3]([2.4,2.4,2.4])) == [4.4,4.4,4.4]) + assert np.all( + transform_collection.transform_point(itk.Point[itk.F, 3]([0, 0, 0])) + == [1, 1, 1] + ) + assert np.all( + transform_collection.transform_point(itk.Point[itk.F, 3]([2.4, 2.4, 2.4])) + == [4.4, 4.4, 4.4] + ) # Transform over overlapping domain region - assert np.all(transform_collection.transform_point(itk.Point[itk.F,3]([1,1,1])) == [2.5,2.5,2.5]) + assert np.all( + transform_collection.transform_point(itk.Point[itk.F, 3]([1, 1, 1])) + == [2.5, 2.5, 2.5] + ) # Transform over excluded domain region fails with pytest.raises(Exception): - print(transform_collection.transform_point([-1,-1,-1])) - -def test_distance_weighted_blending(): - transforms = [TransformEntry(itk.TranslationTransform[itk.D,3].New(), itk.Image[itk.F,3].New()), - TransformEntry(itk.TranslationTransform[itk.D,3].New(), itk.Image[itk.F,3].New())] - transforms[0].transform.Translate([1,1,1]) - transforms[1].transform.Translate([2,2,2]) - transforms[0].domain.SetOrigin([0,0,0]) - transforms[0].domain.SetRegions([4,4,4]) - transforms[1].domain.SetOrigin([1,1,1]) - transforms[1].domain.SetRegions([4,4,4]) - + print(transform_collection.transform_point([-1, -1, -1])) + + +def test_distance_weighted_blending(): + transforms = [ + TransformEntry( + itk.TranslationTransform[itk.D, 3].New(), itk.Image[itk.F, 3].New() + ), + TransformEntry( + itk.TranslationTransform[itk.D, 3].New(), itk.Image[itk.F, 3].New() + ), + ] + transforms[0].transform.Translate([1, 1, 1]) + transforms[1].transform.Translate([2, 2, 2]) + transforms[0].domain.SetOrigin([0, 0, 0]) + transforms[0].domain.SetRegions([4, 4, 4]) + transforms[1].domain.SetOrigin([1, 1, 1]) + transforms[1].domain.SetRegions([4, 4, 4]) + transform_collection = TransformCollection( blend_method=TransformCollection.blend_distance_weighted_mean, - transform_and_domain_list=transforms) + transform_and_domain_list=transforms, + ) assert len(transform_collection.transforms) == 2 assert len(transform_collection.domains) == 2 # Transform over non-overlapping domain region - assert np.all(transform_collection.transform_point(itk.Point[itk.F,3]([0,0,0])) == [1,1,1]) - assert np.all(transform_collection.transform_point(itk.Point[itk.F,3]([4.4,4.4,4.4])) == [6.4,6.4,6.4]) + assert np.all( + transform_collection.transform_point(itk.Point[itk.F, 3]([0, 0, 0])) + == [1, 1, 1] + ) + assert np.all( + transform_collection.transform_point(itk.Point[itk.F, 3]([4.4, 4.4, 4.4])) + == [6.4, 6.4, 6.4] + ) # Transform over excluded domain region fails with pytest.raises(Exception): - print(transform_collection.transform_point([-1,-1,-1])) + print(transform_collection.transform_point([-1, -1, -1])) # Transform over overlapping domain region weights by distance to domain edge - assert np.all(transform_collection.transform_point(itk.Point[itk.F,3]([1,1,1])) == [2.25]*3) - assert np.all(transform_collection.transform_point(itk.Point[itk.F,3]([2,2,2])) == [3.5]*3) - assert np.all(transform_collection.transform_point(itk.Point[itk.F,3]([3,3,3])) == [4.75]*3) - assert np.all(transform_collection.transform_point(itk.Point[itk.F,3]([1,2,3])) == [2.5,3.5,4.5]) + assert np.all( + transform_collection.transform_point(itk.Point[itk.F, 3]([1, 1, 1])) + == [2.25] * 3 + ) + assert np.all( + transform_collection.transform_point(itk.Point[itk.F, 3]([2, 2, 2])) + == [3.5] * 3 + ) + assert np.all( + transform_collection.transform_point(itk.Point[itk.F, 3]([3, 3, 3])) + == [4.75] * 3 + ) + assert np.all( + transform_collection.transform_point(itk.Point[itk.F, 3]([1, 2, 3])) + == [2.5, 3.5, 4.5] + ) diff --git a/src/test/util/mock.py b/src/test/util/mock.py index e04d591..adf8e5b 100644 --- a/src/test/util/mock.py +++ b/src/test/util/mock.py @@ -1,51 +1,59 @@ - from typing import Iterable, Iterator import itk -from itk_dreg.base.image_block_interface import BlockPairRegistrationResult, RegistrationTransformResult, BlockRegStatus, LocatedBlockResult -from itk_dreg.base.registration_interface import BlockPairRegistrationMethod, ReduceResultsMethod +from itk_dreg.base.image_block_interface import ( + BlockPairRegistrationResult, + RegistrationTransformResult, + BlockRegStatus, + LocatedBlockResult, +) +from itk_dreg.base.registration_interface import ( + BlockPairRegistrationMethod, + ReduceResultsMethod, +) -#TODO use unittest.mock +# TODO use unittest.mock class ConstantBlockPairRegistrationMethod(BlockPairRegistrationMethod): """ Return a constant, default registration result for each block. """ - def __init__(self, default_result:BlockPairRegistrationResult=None): - default_transform_domain = itk.Image[itk.F,3].New() - default_transform_domain.SetRegions([10]*3) + + def __init__(self, default_result: BlockPairRegistrationResult = None): + default_transform_domain = itk.Image[itk.F, 3].New() + default_transform_domain.SetRegions([10] * 3) self.default_result = default_result or BlockPairRegistrationResult( - transform=itk.TranslationTransform[itk.D,3].New(), + transform=itk.TranslationTransform[itk.D, 3].New(), transform_domain=default_transform_domain, inv_transform=None, inv_transform_domain=None, - status=BlockRegStatus.SUCCESS + status=BlockRegStatus.SUCCESS, ) def __call__(self, **kwargs): return self.default_result - + + class ConstantReduceResultsMethod(ReduceResultsMethod): """ Return a constant, default transform result. """ - def __init__(self, default_result:RegistrationTransformResult=None): + + def __init__(self, default_result: RegistrationTransformResult = None): self.default_result = default_result or RegistrationTransformResult( - transform=itk.TranslationTransform[itk.D,3].New(), - inv_transform=None + transform=itk.TranslationTransform[itk.D, 3].New(), inv_transform=None ) + def __call__(self, **kwargs): return self.default_result class CountingBlockPairRegistrationMethod(ConstantBlockPairRegistrationMethod): num_calls = 0 - def __call__( - self, - **kwargs - ) -> BlockPairRegistrationResult: + + def __call__(self, **kwargs) -> BlockPairRegistrationResult: self.num_calls += 1 return super().__call__(**kwargs) @@ -53,23 +61,18 @@ def __call__( class CountingReduceResultsMethod(ConstantReduceResultsMethod): num_calls = 0 - def __call__( - self, - **kwargs - ) -> RegistrationTransformResult: + def __call__(self, **kwargs) -> RegistrationTransformResult: self.num_calls += 1 return super().__call__(**kwargs) - + class IteratorBlockPairRegistrationMethod(BlockPairRegistrationMethod): - def __init__(self, default_results:Iterator[BlockPairRegistrationResult]): + def __init__(self, default_results: Iterator[BlockPairRegistrationResult]): self.default_results = default_results def __call__(self, **kwargs): return next(self.default_results) - - class PassthroughReduceResultsMethod(ReduceResultsMethod): """ @@ -77,15 +80,17 @@ class PassthroughReduceResultsMethod(ReduceResultsMethod): May fail if transform inputs are inherently bounded, such as a displacement field. """ - def __init__(self, return_index:int=0): + + def __init__(self, return_index: int = 0): self.return_index = 0 - def __call__(self, block_results: Iterable[LocatedBlockResult],**kwargs): + + def __call__(self, block_results: Iterable[LocatedBlockResult], **kwargs): counter = 0 results_iter = iter(block_results) while counter < self.return_index: next(results_iter) nth_block_result = next(results_iter) return RegistrationTransformResult( - transform=nth_block_result.result.transform, - inv_transform=nth_block_result.result.inv_transform - ) + transform=nth_block_result.result.transform, + inv_transform=nth_block_result.result.inv_transform, + ) From f06a2e70ee178386c42e82c64f09948158e367b4 Mon Sep 17 00:00:00 2001 From: Matt McCormick Date: Mon, 22 Apr 2024 16:46:28 -0400 Subject: [PATCH 7/7] ENH: Simplify CI testing --- .github/workflows/build-test-publish.yml | 25 ++++++++++-------------- 1 file changed, 10 insertions(+), 15 deletions(-) diff --git a/.github/workflows/build-test-publish.yml b/.github/workflows/build-test-publish.yml index 5a1283e..d6a3fbe 100644 --- a/.github/workflows/build-test-publish.yml +++ b/.github/workflows/build-test-publish.yml @@ -13,12 +13,14 @@ jobs: python-version: "3.11" - name: "Build pure Python wheels" run: | - python -m pip wheel -w dist -e src/ + python -m pip install hatch + cd src + hatch build - uses: actions/upload-artifact@v4 with: name: wheel - path: ./dist/itk_dreg-*.whl + path: ./src/dist/itk_dreg-*.whl pytest: name: "Test with PyTest" @@ -36,27 +38,20 @@ jobs: - name: "Install Dependencies" shell: bash + working-directory: src run: | - python -m pip install hatch - cd src - hatch build + python -m pip install -e '.[test]' - - name: Download Python Wheel Artifact - uses: actions/download-artifact@v4 - with: - name: wheel - - - name: "Install itk-dreg from Wheel Artifact" + - name: "Install itk-dreg from source" + working-directory: src run: | wheel_name_full=`(find . -name "itk_dreg-*.whl")` - python -m pip uninstall -y itk_dreg - python -m pip install ${wheel_name_full} + python -m pip install -e '.[test]' - name: "Run Tests" shell: bash + working-directory: src run: | - cd src - pip install -e '.[test]' pytest -vvv -s -k "not localcluster and not serialize_pairwise_result" publish: