From c5e8f50c4769d6b9a81acd8a41eb3c1468d4c2d6 Mon Sep 17 00:00:00 2001 From: Yohsuke Fukai Date: Wed, 25 Oct 2023 09:27:35 +0900 Subject: [PATCH 01/11] solved baseline saving problem --- src/basicpy/basicpy.py | 15 ++++++++++----- tests/test_basic.py | 17 +++++++++++------ 2 files changed, 21 insertions(+), 11 deletions(-) diff --git a/src/basicpy/basicpy.py b/src/basicpy/basicpy.py index 9230921c..9169aa56 100644 --- a/src/basicpy/basicpy.py +++ b/src/basicpy/basicpy.py @@ -74,7 +74,7 @@ class TimelapseTransformMode(str, Enum): _SETTINGS_FNAME = "settings.json" -_PROFILES_FNAME = "profiles.npy" +_PROFILES_FNAME = "profiles.npz" # multiple channels should be handled by creating a `basic` object for each channel @@ -856,8 +856,12 @@ def save_model(self, model_dir: PathLike, overwrite: bool = False) -> None: # NOTE emit warning if profiles are all zeros? fit probably not run # save profiles - profiles = np.array((self.flatfield, self.darkfield)) - np.save(path / _PROFILES_FNAME, profiles) + np.savez( + path / _PROFILES_FNAME, + flatfield=np.array(self.flatfield), + darkfield=np.array(self.darkfield), + baseline=np.array(self.baseline), + ) @classmethod def load_model(cls, model_dir: PathLike) -> BaSiC: @@ -871,7 +875,8 @@ def load_model(cls, model_dir: PathLike) -> BaSiC: model = json.load(fp) profiles = np.load(path / _PROFILES_FNAME) - model["flatfield"] = profiles[0] - model["darkfield"] = profiles[1] + model["flatfield"] = profiles["flatfield"] + model["darkfield"] = profiles["darkfield"] + model["baseline"] = profiles["baseline"] return BaSiC(**model) diff --git a/tests/test_basic.py b/tests/test_basic.py index 1f388db5..26b3d855 100644 --- a/tests/test_basic.py +++ b/tests/test_basic.py @@ -273,6 +273,7 @@ def basic_object(request): # set profiles basic.flatfield = np.full((128,) * dim, 1, dtype=np.float64) basic.darkfield = np.full((128,) * dim, 2, dtype=np.float64) + basic.baseline = np.ones(100, dtype=np.float64) return basic @@ -284,18 +285,19 @@ def test_basic_save_model(tmp_path: Path, basic_object): # check that the files exists assert (model_dir / "settings.json").exists() - assert (model_dir / "profiles.npy").exists() + assert (model_dir / "profiles.npz").exists() # load files and check for expected content - saved_profiles = np.load(model_dir / "profiles.npy") - profiles = np.array((basic_object.flatfield, basic_object.darkfield)) - assert np.array_equal(saved_profiles, profiles) + saved_profiles = np.load(model_dir / "profiles.npz") + assert np.array_equal(saved_profiles["flatfield"], basic_object.flatfield) + assert np.array_equal(saved_profiles["darkfield"], basic_object.darkfield) + assert np.array_equal(saved_profiles["baseline"], basic_object.baseline) # TODO check settings contents # remove files but not the folder to check for overwriting (model_dir / "settings.json").unlink() - (model_dir / "profiles.npy").unlink() + (model_dir / "profiles.npz").unlink() # assert not (model_dir / "settings.json").exists() # assert not (model_dir / "profiles.npy").exists() @@ -306,7 +308,7 @@ def test_basic_save_model(tmp_path: Path, basic_object): # overwrites if specified basic_object.save_model(model_dir, overwrite=True) assert (model_dir / "settings.json").exists() - assert (model_dir / "profiles.npy").exists() + assert (model_dir / "profiles.npz").exists() def test_basic_save_load_model(tmp_path: Path, basic_object): @@ -322,6 +324,9 @@ def test_basic_save_load_model(tmp_path: Path, basic_object): assert np.allclose(basic2.darkfield, darkfield) assert basic_object.dict() == basic2.dict() + images = datasets.wsi_brain() + basic_object.fit(images) + @pytest.fixture def profiles(): From 0834cb957eab4a25308b279b8e48b02dfdd5c9c2 Mon Sep 17 00:00:00 2001 From: Yohsuke Fukai Date: Wed, 25 Oct 2023 10:07:35 +0900 Subject: [PATCH 02/11] fixed load model test --- tests/test_basic.py | 18 ++++++++++++------ 1 file changed, 12 insertions(+), 6 deletions(-) diff --git a/tests/test_basic.py b/tests/test_basic.py index 26b3d855..48dedcae 100644 --- a/tests/test_basic.py +++ b/tests/test_basic.py @@ -331,11 +331,11 @@ def test_basic_save_load_model(tmp_path: Path, basic_object): @pytest.fixture def profiles(): # create and write mock profiles to file - profiles = np.zeros((2, 128, 128), dtype=np.float64) + flatfield = np.zeros((128, 128), dtype=np.float64) + darkfield = np.zeros((128, 128), dtype=np.float64) # unique profiles to check that they are in proper place - profiles[0] = 1 - profiles[1] = 2 - return profiles + baseline = np.ones(100, dtype=np.float64) + return flatfield, darkfield, baseline @pytest.fixture @@ -348,12 +348,17 @@ def model_path(tmp_path, profiles): """ with open(tmp_path / "settings.json", "w") as fp: fp.write(settings_json) - np.save(tmp_path / "profiles.npy", profiles) + np.savez( + tmp_path / "profiles.npz", + flatfield=profiles[0], + darkfield=profiles[1], + baseline=profiles[2], + ) return str(tmp_path) @pytest.mark.parametrize("raises_error", [(True), (False)], ids=["no_model", "model"]) -def test_basic_load_model(model_path: str, raises_error: bool, profiles: np.ndarray): +def test_basic_load_model(model_path: str, raises_error: bool, profiles): if raises_error: with pytest.raises(FileNotFoundError): basic = BaSiC.load_model("/not/a/real/path") @@ -367,6 +372,7 @@ def test_basic_load_model(model_path: str, raises_error: bool, profiles: np.ndar # check that the profiles are in the right places assert np.array_equal(basic.flatfield, profiles[0]) assert np.array_equal(basic.darkfield, profiles[1]) + assert np.array_equal(basic.baseline, profiles[2]) # check that settings are not default assert basic.epsilon != BaSiC.__fields__["epsilon"].default From dd645818a6c3bfcefd2cf071e13e7fb69d73109f Mon Sep 17 00:00:00 2001 From: Yohsuke Fukai Date: Mon, 13 Nov 2023 15:11:11 +0900 Subject: [PATCH 03/11] updated definition of fourier_L0_norm --- src/basicpy/metrics.py | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/src/basicpy/metrics.py b/src/basicpy/metrics.py index 17cb2176..3174db3b 100644 --- a/src/basicpy/metrics.py +++ b/src/basicpy/metrics.py @@ -60,10 +60,13 @@ def fourier_L0_norm( image: np.ndarray, threshold: float = 0.1, fourier_radius: float = 10, + exclude_edges: bool = True, ): SF = dctn(image) xy = np.meshgrid(*[range(x) for x in image.shape], indexing="ij") - outside_radius = np.sum(np.array(xy) ** 2, axis=0) > fourier_radius**2 + outside_radius = (np.sum(np.array(xy) ** 2, axis=0) > fourier_radius**2) + if exclude_edges: + outside_radius = outside_radius & (xy[0] > 0) & (xy[1] > 0) L0_norm = np.sum(SF[outside_radius] > threshold) / np.sum(outside_radius) return L0_norm @@ -117,6 +120,7 @@ def autotune_cost( flatfield, fourier_l0_norm_image_threshold, fourier_l0_norm_fourier_radius, + exclude_edges=True, ) if n < fourier_l0_norm_threshold: From 7b02537943595a893c45777a9426b756b9f47da3 Mon Sep 17 00:00:00 2001 From: "Yohsuke T. Fukai" Date: Tue, 5 Dec 2023 19:50:07 +0900 Subject: [PATCH 04/11] Create codeofconduct.rst Added code of conduct content in the doc --- docs/codeofconduct.rst | 1 + 1 file changed, 1 insertion(+) create mode 100644 docs/codeofconduct.rst diff --git a/docs/codeofconduct.rst b/docs/codeofconduct.rst new file mode 100644 index 00000000..96e0ba2f --- /dev/null +++ b/docs/codeofconduct.rst @@ -0,0 +1 @@ +.. include:: ../CODE_OF_CONDUCT.rst From 80e5a8369bf5c6d9eacb0d2de5d71649f0752250 Mon Sep 17 00:00:00 2001 From: Yohsuke Fukai Date: Tue, 12 Dec 2023 23:19:07 +0900 Subject: [PATCH 05/11] updated default value for autotune --- src/basicpy/basicpy.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/basicpy/basicpy.py b/src/basicpy/basicpy.py index 9230921c..3b0a126b 100644 --- a/src/basicpy/basicpy.py +++ b/src/basicpy/basicpy.py @@ -658,8 +658,8 @@ def autotune( histogram_use_fitting_weight: bool = True, fourier_l0_norm_image_threshold: float = 0.1, fourier_l0_norm_fourier_radius=10, - fourier_l0_norm_threshold=1e-3, - fourier_l0_norm_cost_coef=1e4, + fourier_l0_norm_threshold=0.0, + fourier_l0_norm_cost_coef=30, early_stop: bool = True, early_stop_n_iter_no_change: int = 15, early_stop_torelance: float = 1e-6, From 41aaeaddfa088ca5c91d5da84dac0b8f7c579d36 Mon Sep 17 00:00:00 2001 From: Yohsuke Fukai Date: Tue, 12 Dec 2023 23:32:24 +0900 Subject: [PATCH 06/11] updated test for windows --- tests/test_basic.py | 5 ++--- 1 file changed, 2 insertions(+), 3 deletions(-) diff --git a/tests/test_basic.py b/tests/test_basic.py index 48dedcae..2fabc29a 100644 --- a/tests/test_basic.py +++ b/tests/test_basic.py @@ -277,9 +277,8 @@ def basic_object(request): return basic -def test_basic_save_model(tmp_path: Path, basic_object): - model_dir = tmp_path / "test_model" - +def test_basic_save_model(tmp_path_factory, basic_object): + model_dir = Path(tmp_path_factory.mktemp("data")) / "test_model" # save the model basic_object.save_model(model_dir) From f82b5df53312c9a1fbb3d59d0dd1fb670daa506e Mon Sep 17 00:00:00 2001 From: Yohsuke Fukai Date: Tue, 12 Dec 2023 23:34:51 +0900 Subject: [PATCH 07/11] updated tmp_path --- tests/test_basic.py | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/tests/test_basic.py b/tests/test_basic.py index 2fabc29a..e6a551a2 100644 --- a/tests/test_basic.py +++ b/tests/test_basic.py @@ -310,7 +310,8 @@ def test_basic_save_model(tmp_path_factory, basic_object): assert (model_dir / "profiles.npz").exists() -def test_basic_save_load_model(tmp_path: Path, basic_object): +def test_basic_save_load_model(tmp_path_factory, basic_object): + tmp_path = Path(tmp_path_factory.mktemp("data")) model_dir = tmp_path / "test_model" flatfield = basic_object.flatfield.copy() darkfield = basic_object.darkfield.copy() @@ -338,7 +339,8 @@ def profiles(): @pytest.fixture -def model_path(tmp_path, profiles): +def model_path(tmp_path_factory, profiles): + tmp_path = Path(tmp_path_factory.mktemp("data")) settings_json = """\ {"epsilon": 0.2, "get_darkfield": false, "smoothness_darkfield": 0.0, "smoothness_flatfield": 0.0, "max_iterations": 500, From de3d8bc8f56da67a1f97a06ee042f0c7a336089a Mon Sep 17 00:00:00 2001 From: Yohsuke Fukai Date: Sat, 16 Dec 2023 14:21:42 +0900 Subject: [PATCH 08/11] updated test_basic_save_model so that the directory name will be unique --- tests/test_basic.py | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/tests/test_basic.py b/tests/test_basic.py index e6a551a2..1d2dde9d 100644 --- a/tests/test_basic.py +++ b/tests/test_basic.py @@ -33,7 +33,7 @@ def synthesized_test_data(request): grid = np.array( np.meshgrid( *[np.linspace(-size // 2 + 1, size // 2, size) for size in sizes], - indexing="ij" + indexing="ij", ) ) @@ -278,7 +278,8 @@ def basic_object(request): def test_basic_save_model(tmp_path_factory, basic_object): - model_dir = Path(tmp_path_factory.mktemp("data")) / "test_model" + dim = basic_object.flatfield.ndim + model_dir = Path(tmp_path_factory.mktemp("data")) / f"test_save_model_{dim}" # save the model basic_object.save_model(model_dir) From 75bf2ab9c19587f3c05e755f06432123fcf4a22a Mon Sep 17 00:00:00 2001 From: Yohsuke Fukai Date: Sat, 16 Dec 2023 15:26:38 +0900 Subject: [PATCH 09/11] further update --- tests/test_basic.py | 15 +++++++++++++-- 1 file changed, 13 insertions(+), 2 deletions(-) diff --git a/tests/test_basic.py b/tests/test_basic.py index 1d2dde9d..73f92a94 100644 --- a/tests/test_basic.py +++ b/tests/test_basic.py @@ -1,4 +1,5 @@ from pathlib import Path +from time import sleep import numpy as np import pytest @@ -296,8 +297,18 @@ def test_basic_save_model(tmp_path_factory, basic_object): # TODO check settings contents # remove files but not the folder to check for overwriting - (model_dir / "settings.json").unlink() - (model_dir / "profiles.npz").unlink() + for filename in ["settings.json", "profiles.npz"]: + i = 0 + while True: + try: + (model_dir / filename).unlink() + break + except PermissionError as e: + sleep(1) + i += 1 + if i > 5: + raise e + # assert not (model_dir / "settings.json").exists() # assert not (model_dir / "profiles.npy").exists() From 876d23c0fc333916aafbf1fde069bb4e9d9837bd Mon Sep 17 00:00:00 2001 From: Yohsuke Fukai Date: Sat, 16 Dec 2023 15:56:15 +0900 Subject: [PATCH 10/11] gave up deleting files but instead creating new folder --- tests/test_basic.py | 21 +++------------------ 1 file changed, 3 insertions(+), 18 deletions(-) diff --git a/tests/test_basic.py b/tests/test_basic.py index 73f92a94..d33d1065 100644 --- a/tests/test_basic.py +++ b/tests/test_basic.py @@ -1,5 +1,4 @@ from pathlib import Path -from time import sleep import numpy as np import pytest @@ -296,25 +295,11 @@ def test_basic_save_model(tmp_path_factory, basic_object): # TODO check settings contents - # remove files but not the folder to check for overwriting - for filename in ["settings.json", "profiles.npz"]: - i = 0 - while True: - try: - (model_dir / filename).unlink() - break - except PermissionError as e: - sleep(1) - i += 1 - if i > 5: - raise e - - # assert not (model_dir / "settings.json").exists() - # assert not (model_dir / "profiles.npy").exists() - + model_dir2 = Path(tmp_path_factory.mktemp("data")) / f"test_save_model_{dim}_2" + model_dir2.mkdir() # an error raises when the model folder exists with pytest.raises(FileExistsError): - basic_object.save_model(model_dir) + basic_object.save_model(model_dir2) # overwrites if specified basic_object.save_model(model_dir, overwrite=True) From efd4e8daeda0aa2cf55dcca20e70f06bfc19e62a Mon Sep 17 00:00:00 2001 From: Yohsuke Fukai Date: Sat, 16 Dec 2023 15:56:50 +0900 Subject: [PATCH 11/11] added new case --- tests/test_basic.py | 3 +++ 1 file changed, 3 insertions(+) diff --git a/tests/test_basic.py b/tests/test_basic.py index d33d1065..1b773faf 100644 --- a/tests/test_basic.py +++ b/tests/test_basic.py @@ -305,6 +305,9 @@ def test_basic_save_model(tmp_path_factory, basic_object): basic_object.save_model(model_dir, overwrite=True) assert (model_dir / "settings.json").exists() assert (model_dir / "profiles.npz").exists() + basic_object.save_model(model_dir2, overwrite=True) + assert (model_dir2 / "settings.json").exists() + assert (model_dir2 / "profiles.npz").exists() def test_basic_save_load_model(tmp_path_factory, basic_object):