Exemple #1
0
def to_samples(sample_list, data_io=None, load_seg=None, load_pred=None):

    if load_seg is None:
        seg = True
    else:
        seg = load_seg

    if load_pred is None:
        pred = True
    else:
        pred = load_pred

    res = []

    for sample in sample_list:
        if isinstance(sample, Sample):
            if load_seg is None:
                seg &= sample.seg_data is not None
            if load_pred is None:
                pred &= sample.pred_data is not None
            res.append(sample)
            continue
        elif isinstance(sample, str):
            res.append(load_samples([sample], data_io, seg, pred))
        elif isinstance(sample, np.ndarray):
            sampleObj = Sample(
                "ndarray_" +
                str(random.choice(range(999999999)), sample, 1, 0))
            sampleObj.img_data = sample
            res.append(sampleObj)
        else:
            raise ValueError("Cannot interpret an object of type " +
                             str(type(sample)) + " as a sample")

    return res
Exemple #2
0
 def setUpClass(self):
     np.random.seed(1234)
     # Create imgaging and segmentation data
     img2D = np.random.rand(16, 16) * 255
     img2D = img2D.astype(int)
     img3D = np.random.rand(16, 16, 16) * 255
     img3D = img3D.astype(int)
     seg2D = np.random.rand(16, 16) * 3
     seg2D = seg2D.astype(int)
     seg3D = np.random.rand(16, 16, 16) * 3
     seg3D = seg3D.astype(int)
     # Create testing samples
     self.sample2D = Sample("sample2D", img2D, channels=1, classes=3)
     self.sample2Dseg = Sample("sample2Dseg", img2D, channels=1, classes=3)
     self.sample3D = Sample("sample3D", img3D, channels=1, classes=3)
     self.sample3Dseg = Sample("sample3Dseg", img3D, channels=1, classes=3)
     # Add segmentation to seg samples
     self.sample2Dseg.add_segmentation(seg2D)
     self.sample3Dseg.add_segmentation(seg3D)
Exemple #3
0
class SubfunctionsTEST(unittest.TestCase):
    # Create random imaging and segmentation data
    @classmethod
    def setUpClass(self):
        np.random.seed(1234)
        # Create imgaging and segmentation data
        img2D = np.random.rand(16, 16) * 255
        img2D = img2D.astype(int)
        img3D = np.random.rand(16, 16, 16) * 255
        img3D = img3D.astype(int)
        seg2D = np.random.rand(16, 16) * 3
        seg2D = seg2D.astype(int)
        seg3D = np.random.rand(16, 16, 16) * 3
        seg3D = seg3D.astype(int)
        # Create testing samples
        self.sample2D = Sample("sample2D", img2D, channels=1, classes=3)
        self.sample2Dseg = Sample("sample2Dseg", img2D, channels=1, classes=3)
        self.sample3D = Sample("sample3D", img3D, channels=1, classes=3)
        self.sample3Dseg = Sample("sample3Dseg", img3D, channels=1, classes=3)
        # Add segmentation to seg samples
        self.sample2Dseg.add_segmentation(seg2D)
        self.sample3Dseg.add_segmentation(seg3D)

    #-------------------------------------------------#
    #                Base Functionality               #
    #-------------------------------------------------#
    # Run multiple or none subfunctions during preprocessing
    def test_SUBFUNCTIONS_preprocessing(self):
        ds = dict()
        for i in range(0, 10):
            img = np.random.rand(16, 16, 16) * 255
            img = img.astype(int)
            seg = np.random.rand(16, 16, 16) * 3
            seg = seg.astype(int)
            sample = (img, seg)
            ds["TEST.sample_" + str(i)] = sample
        io_interface = Dictionary_interface(ds, classes=3, three_dim=True)
        self.tmp_dir = tempfile.TemporaryDirectory(prefix="tmp.miscnn.")
        tmp_batches = os.path.join(self.tmp_dir.name, "batches")
        dataio = Data_IO(io_interface, input_path="", output_path="",
                         batch_path=tmp_batches, delete_batchDir=False)
        sf = [Resize((8,8,8)), Normalization(), Clipping(min=-1.0, max=0.0)]
        pp = Preprocessor(dataio, data_aug=None, batch_size=1,
                          prepare_subfunctions=False, analysis="fullimage",
                          subfunctions=sf)
        sample_list = dataio.get_indiceslist()
        batches = pp.run(sample_list, training=True, validation=False)
        for i in range(0, 10):
            img = batches[i][0]
            seg = batches[i][1]
            self.assertEqual(img.shape, (1,8,8,8,1))
            self.assertEqual(seg.shape, (1,8,8,8,3))
            self.assertTrue(np.min(img) >= -1.0 and np.max(img) <= 0.0)
        self.tmp_dir.cleanup()

    # Run multiple or none subfunctions during postprocessing
    def test_SUBFUNCTIONS_postprocessing(self):
        ds = dict()
        for i in range(0, 10):
            img = np.random.rand(16, 16, 16) * 255
            img = img.astype(int)
            seg = np.random.rand(16, 16, 16) * 3
            seg = seg.astype(int)
            sample = (img, seg)
            ds["TEST.sample_" + str(i)] = sample
        io_interface = Dictionary_interface(ds, classes=3, three_dim=True)
        self.tmp_dir = tempfile.TemporaryDirectory(prefix="tmp.miscnn.")
        tmp_batches = os.path.join(self.tmp_dir.name, "batches")
        dataio = Data_IO(io_interface, input_path="", output_path="",
                         batch_path=tmp_batches, delete_batchDir=False)
        sf = [Resize((9,9,9)), Normalization(), Clipping(min=-1.0, max=0.0)]
        pp = Preprocessor(dataio, batch_size=1, prepare_subfunctions=False,
                          analysis="patchwise-grid", subfunctions=sf,
                          patch_shape=(4,4,4))
        sample_list = dataio.get_indiceslist()
        for index in sample_list:
            sample = dataio.sample_loader(index)
            for sf in pp.subfunctions:
                sf.preprocessing(sample, training=False)
            pp.cache["shape_" + str(index)] = sample.img_data.shape
            sample.seg_data = np.random.rand(9, 9, 9) * 3
            sample.seg_data = sample.seg_data.astype(int)
            sample.seg_data = to_categorical(sample.seg_data, num_classes=3)
            data_patches = pp.analysis_patchwise_grid(sample, training=True,
                                                      data_aug=False)
            seg_list = []
            for i in range(0, len(data_patches)):
                seg_list.append(data_patches[i][1])
            seg = np.stack(seg_list, axis=0)
            self.assertEqual(seg.shape, (27,4,4,4,3))
            pred = pp.postprocessing(sample, seg)
            self.assertEqual(pred.shape, (16,16,16))
        self.tmp_dir.cleanup()

    # Run prepare subfunction of Preprocessor
    def test_SUBFUNCTIONS_prepare(self):
        ds = dict()
        for i in range(0, 10):
            img = np.random.rand(16, 16, 16) * 255
            img = img.astype(int)
            seg = np.random.rand(16, 16, 16) * 3
            seg = seg.astype(int)
            sample = (img, seg)
            ds["TEST.sample_" + str(i)] = sample
        io_interface = Dictionary_interface(ds, classes=3, three_dim=True)
        self.tmp_dir = tempfile.TemporaryDirectory(prefix="tmp.miscnn.")
        tmp_batches = os.path.join(self.tmp_dir.name, "batches")
        dataio = Data_IO(io_interface, input_path="", output_path="",
                         batch_path=tmp_batches, delete_batchDir=False)
        sf = [Resize((8,8,8)), Normalization(), Clipping(min=-1.0, max=0.0)]
        pp = Preprocessor(dataio, batch_size=1, prepare_subfunctions=True,
                          analysis="fullimage", subfunctions=sf)
        sample_list = dataio.get_indiceslist()
        pp.run_subfunctions(sample_list, training=True)
        batches = pp.run(sample_list, training=True, validation=False)
        self.assertEqual(len(os.listdir(tmp_batches)), 10)
        for i in range(0, 10):
            file_prepared_subfunctions = os.path.join(tmp_batches,
                    str(pp.data_io.seed) + ".TEST.sample_" + str(i) + ".pickle")
            self.assertTrue(os.path.exists(file_prepared_subfunctions))
            img = batches[i][0]
            seg = batches[i][1]
            self.assertIsNotNone(img)
            self.assertIsNotNone(seg)
            self.assertEqual(img.shape, (1,8,8,8,1))
            self.assertEqual(seg.shape, (1,8,8,8,3))
        self.tmp_dir.cleanup()

    # Run prepare subfunction of Preprocessor using multi-processing
    def test_SUBFUNCTIONS_prepare_MULTIPROCESSING(self):
        ds = dict()
        for i in range(0, 5):
            img = np.random.rand(16, 16, 16) * 255
            img = img.astype(int)
            seg = np.random.rand(16, 16, 16) * 3
            seg = seg.astype(int)
            sample = (img, seg)
            ds["TEST.sample_" + str(i)] = sample
        io_interface = Dictionary_interface(ds, classes=3, three_dim=True)
        self.tmp_dir = tempfile.TemporaryDirectory(prefix="tmp.miscnn.")
        tmp_batches = os.path.join(self.tmp_dir.name, "batches")
        dataio = Data_IO(io_interface, input_path="", output_path="",
                         batch_path=tmp_batches, delete_batchDir=False)
        sf = [Resize((8,8,8)), Normalization(), Clipping(min=-1.0, max=0.0)]
        pp = Preprocessor(dataio, batch_size=1, prepare_subfunctions=True,
                          analysis="fullimage", subfunctions=sf,
                          use_multiprocessing=True)
        pp.mp_threads = 4
        sample_list = dataio.get_indiceslist()
        pp.run_subfunctions(sample_list, training=True)
        batches = pp.run(sample_list, training=True, validation=False)
        self.assertEqual(len(os.listdir(tmp_batches)), 5)
        for i in range(0, 5):
            file_prepared_subfunctions = os.path.join(tmp_batches,
                    str(pp.data_io.seed) + ".TEST.sample_" + str(i) + ".pickle")
            self.assertTrue(os.path.exists(file_prepared_subfunctions))
            img = batches[i][0]
            seg = batches[i][1]
            self.assertIsNotNone(img)
            self.assertIsNotNone(seg)
            self.assertEqual(img.shape, (1,8,8,8,1))
            self.assertEqual(seg.shape, (1,8,8,8,3))
        #self.tmp_dir.cleanup()

    # Run prepare subfunction of Preprocessor
    def test_SUBFUNCTIONS_fullrun(self):
        ds = dict()
        for i in range(0, 10):
            img = np.random.rand(16, 16, 16) * 255
            img = img.astype(int)
            seg = np.random.rand(16, 16, 16) * 3
            seg = seg.astype(int)
            sample = (img, seg)
            ds["TEST.sample_" + str(i)] = sample
        io_interface = Dictionary_interface(ds, classes=3, three_dim=True)
        self.tmp_dir = tempfile.TemporaryDirectory(prefix="tmp.miscnn.")
        tmp_batches = os.path.join(self.tmp_dir.name, "batches")
        dataio = Data_IO(io_interface, input_path="", output_path="",
                         batch_path=tmp_batches, delete_batchDir=False)
        sf = [Resize((16,16,16)), Normalization(), Clipping(min=-1.0, max=0.0)]
        pp = Preprocessor(dataio, batch_size=1, prepare_subfunctions=True,
                          analysis="fullimage", subfunctions=sf)
        nn = Neural_Network(preprocessor=pp)
        sample_list = dataio.get_indiceslist()
        nn.predict(sample_list, return_output=True)

    #-------------------------------------------------#
    #                    Resizing                     #
    #-------------------------------------------------#
    def test_SUBFUNCTIONS_RESIZE_preprocessing(self):
        # Test for 2D and 3D
        for dim in ["2D", "3D"]:
            # Initialize Subfunction
            if dim == "2D" : new_shape = (7,7)
            else : new_shape = (7,7,7)
            sf = Resize(new_shape=new_shape)
            # Test for training as well as prediction
            for train in [True, False]:
                # Create sample object from template
                varname = "sample" + dim
                if train : varname += "seg"
                sample = deepcopy(getattr(self, varname))
                # Run preprocessing of the subfunction
                sf.preprocessing(sample, training=train)
                # Check for correctness
                self.assertEqual(sample.img_data.shape, new_shape + (1,))
                if train : self.assertEqual(sample.seg_data.shape,
                                            new_shape + (1,))

    def test_SUBFUNCTIONS_RESIZE_postprocessing(self):
        # Test for 2D and 3D
        for dim in ["2D", "3D"]:
            # Initialize Subfunction
            if dim == "2D" : new_shape = (7,7)
            else : new_shape = (7,7,7)
            sf = Resize(new_shape=new_shape)
            # Create sample objects
            sample_pred = deepcopy(getattr(self, "sample" + dim))
            sample_train = deepcopy(getattr(self, "sample" + dim + "seg"))
            # Run preprocessing of the subfunction
            sf.preprocessing(sample_train, training=True)
            sf.preprocessing(sample_pred, training=False)
            # Transform segmentation data to simulate prediction data
            sample_pred.pred_data = np.squeeze(sample_train.seg_data, axis=-1)
            # Run postprocessing of the subfunction
            pred = sf.postprocessing(sample_pred, sample_pred.pred_data)
            # Check for correctness
            if dim == "2D" : old_shape = (16,16)
            else : old_shape = (16,16,16)
            self.assertEqual(pred.shape, old_shape)

    def test_SUBFUNCTIONS_RESIZE_postprocessing_activationOutput(self):
        # Test for 2D and 3D
        for dim in ["2D", "3D"]:
            # Initialize Subfunction
            if dim == "2D" : new_shape = (7,7)
            else : new_shape = (7,7,7)
            sf = Resize(new_shape=new_shape)
            # Create sample objects
            sample_pred = deepcopy(getattr(self, "sample" + dim))
            sample_train = deepcopy(getattr(self, "sample" + dim + "seg"))
            # Run preprocessing of the subfunction
            sf.preprocessing(sample_train, training=True)
            sf.preprocessing(sample_pred, training=False)
            # Transform segmentation data to simulate prediction data
            if dim == "2D":
                sample_pred.pred_data = np.random.rand(16, 16, 3)
            else:
                sample_pred.pred_data = np.random.rand(16, 16, 16, 3)
            # Run postprocessing of the subfunction
            pred = sf.postprocessing(sample_pred, sample_pred.pred_data,
                                     activation_output=True)
            # Check for correctness
            if dim == "2D" : old_shape = (16,16,3)
            else : old_shape = (16,16,16,3)
            self.assertEqual(pred.shape, old_shape)

    #-------------------------------------------------#
    #                   Resampling                    #
    #-------------------------------------------------#
    def test_SUBFUNCTIONS_RESAMPLING_preprocessing(self):
        # Test for 2D and 3D
        for dim in ["2D", "3D"]:
            # Initialize Subfunction
            if dim == "2D" : spacing = (1,1)
            else : spacing = (1,1,1)
            sf = Resampling(new_spacing=spacing)
            # Test for training as well as prediction
            for train in [True, False]:
                # Create sample object from template
                varname = "sample" + dim
                if train : varname += "seg"
                sample = deepcopy(getattr(self, varname))
                if dim == "2D" : old_spacing = (1.8, 3.0)
                else : old_spacing = (1.8, 3.0, 3.0)
                sample.extended = {"spacing":np.array(old_spacing)}
                # Run preprocessing of the subfunction
                sf.preprocessing(sample, training=train)
                # Check for correctness
                if dim == "2D":
                    self.assertEqual(sample.img_data.shape, ((28, 48, 1)))
                    if train:
                        self.assertEqual(sample.seg_data.shape, ((28, 48, 1)))
                else:
                    self.assertEqual(sample.img_data.shape, ((28, 48, 48, 1)))
                    if train:
                        self.assertEqual(sample.seg_data.shape,
                                         ((28, 48, 48, 1)))

    def test_SUBFUNCTIONS_RESAMPLING_postprocessing(self):
        # Test for 2D and 3D
        for dim in ["2D", "3D"]:
            # Initialize Subfunction
            if dim == "2D" : spacing = (1,1)
            else : spacing = (1,1,1)
            sf = Resampling(new_spacing=spacing)
            # Create sample objects
            sample_pred = deepcopy(getattr(self, "sample" + dim))
            sample_train = deepcopy(getattr(self, "sample" + dim + "seg"))
            if dim == "2D" : old_spacing = (1.8, 3.0)
            else : old_spacing = (1.8, 3.0, 3.0)
            sample_pred.extended = {"spacing":np.array(old_spacing)}
            sample_train.extended = {"spacing":np.array(old_spacing)}
            # Run preprocessing of the subfunction
            sf.preprocessing(sample_train, training=True)
            sf.preprocessing(sample_pred, training=False)
            # Transform segmentation data to simulate prediction data
            sample_pred.pred_data = np.squeeze(sample_train.seg_data, axis=-1)
            # Run postprocessing of the subfunction
            pred = sf.postprocessing(sample_pred, sample_pred.pred_data)
            # Check for correctness
            if dim == "2D" : old_shape = (16,16)
            else : old_shape = (16,16,16)
            self.assertEqual(pred.shape, old_shape)

    def test_SUBFUNCTIONS_RESAMPLING_postprocessing_activationOutput(self):
        # Test for 2D and 3D
        for dim in ["2D", "3D"]:
            # Initialize Subfunction
            if dim == "2D" : spacing = (1,1)
            else : spacing = (1,1,1)
            sf = Resampling(new_spacing=spacing)
            # Create sample objects
            sample_pred = deepcopy(getattr(self, "sample" + dim))
            sample_train = deepcopy(getattr(self, "sample" + dim + "seg"))
            if dim == "2D" : old_spacing = (1.8, 3.0)
            else : old_spacing = (1.8, 3.0, 3.0)
            sample_pred.extended = {"spacing":np.array(old_spacing)}
            sample_train.extended = {"spacing":np.array(old_spacing)}
            # Run preprocessing of the subfunction
            sf.preprocessing(sample_train, training=True)
            sf.preprocessing(sample_pred, training=False)
            # Transform segmentation data to simulate prediction data
            if dim == "2D":
                sample_pred.pred_data = np.random.rand(16, 16, 3)
            else:
                sample_pred.pred_data = np.random.rand(16, 16, 16, 3)
            # Run postprocessing of the subfunction
            pred = sf.postprocessing(sample_pred, sample_pred.pred_data,
                                     activation_output=True)
            # Check for correctness
            if dim == "2D" : old_shape = (16,16,3)
            else : old_shape = (16,16,16,3)
            self.assertEqual(pred.shape, old_shape)

    #-------------------------------------------------#
    #                     Padding                     #
    #-------------------------------------------------#
    def test_SUBFUNCTIONS_PADDING_preprocessing(self):
        # Test for 2D and 3D
        for dim in ["2D", "3D"]:
            # Initialize Subfunction
            if dim == "2D" : size = (32,8)
            else : size = (8,32,20)
            sf = Padding(min_size=size)
            # Test for training as well as prediction
            for train in [True, False]:
                # Create sample object from template
                varname = "sample" + dim
                if train : varname += "seg"
                sample = deepcopy(getattr(self, varname))
                # Run preprocessing of the subfunction
                sf.preprocessing(sample, training=train)
                # Check for correctness
                if dim == "2D" : new_shape = (32,16)
                else : new_shape = (16,32,20)
                self.assertEqual(sample.img_data.shape, new_shape + (1,))
                if train : self.assertEqual(sample.seg_data.shape,
                                            new_shape + (1,))

    def test_SUBFUNCTIONS_PADDING_postprocessing(self):
        # Test for 2D and 3D
        for dim in ["2D", "3D"]:
            # Initialize Subfunction
            if dim == "2D" : size = (32,8)
            else : size = (8,32,20)
            sf = Padding(min_size=size)
            # Create sample objects
            sample_pred = deepcopy(getattr(self, "sample" + dim))
            sample_train = deepcopy(getattr(self, "sample" + dim + "seg"))
            # Run preprocessing of the subfunction
            sf.preprocessing(sample_train, training=True)
            sf.preprocessing(sample_pred, training=False)
            # Transform segmentation data to simulate prediction data
            sample_pred.pred_data = np.squeeze(sample_train.seg_data, axis=-1)
            # Run postprocessing of the subfunction
            pred = sf.postprocessing(sample_pred, sample_pred.pred_data)
            # Check for correctness
            if dim == "2D" : old_shape = (16,16)
            else : old_shape = (16,16,16)
            self.assertEqual(pred.shape, old_shape)

    #-------------------------------------------------#
    #                    Clipping                     #
    #-------------------------------------------------#
    def test_SUBFUNCTIONS_CLIPPING_preprocessing(self):
        # Initialize Subfunction
        sf = Clipping(min=5, max=10)
        # Test for 2D and 3D
        for dim in ["2D", "3D"]:
            # Test for training as well as prediction
            for train in [True, False]:
                # Create sample object from template
                varname = "sample" + dim
                if train : varname += "seg"
                sample = deepcopy(getattr(self, varname))
                # Run preprocessing of the subfunction
                sf.preprocessing(sample, training=train)
                # Check for correctness
                self.assertTrue(np.array_equal(sample.seg_data,
                                getattr(self, varname).seg_data))
                self.assertEqual(np.min(sample.img_data), 5)
                self.assertEqual(np.max(sample.img_data), 10)

    def test_SUBFUNCTIONS_CLIPPING_postprocessing(self):
        # Initialize Subfunction
        sf = Clipping(min=5, max=10)
        # Test for 2D and 3D
        for dim in ["2D", "3D"]:
            # Create sample objects
            sample_pred = deepcopy(getattr(self, "sample" + dim))
            sample_train = deepcopy(getattr(self, "sample" + dim + "seg"))
            # Run preprocessing of the subfunction
            sf.preprocessing(sample_train, training=True)
            sf.preprocessing(sample_pred, training=False)
            # Transform segmentation data to simulate prediction data
            sample_pred.pred_data = np.squeeze(sample_train.seg_data, axis=-1)
            # Run postprocessing of the subfunction
            pred = sf.postprocessing(sample_pred, sample_pred.pred_data)
            # Check for correctness
            self.assertTrue(np.array_equal(pred, sample_pred.pred_data))

    #-------------------------------------------------#
    #                  Normalization                  #
    #-------------------------------------------------#
    def test_SUBFUNCTIONS_NORMALIZATION_preprocessing(self):
        param_list = ["z-score", "minmax", "grayscale"]
        for param in param_list:
            # Initialize Subfunction
            sf = Normalization(mode=param)
            # Test for 2D and 3D
            for dim in ["2D", "3D"]:
                # Test for training as well as prediction
                for train in [True, False]:
                    # Create sample object from template
                    varname = "sample" + dim
                    if train : varname += "seg"
                    sample = deepcopy(getattr(self, varname))
                    # Run preprocessing of the subfunction
                    sf.preprocessing(sample, training=train)
                    # Check for correctness
                    self.assertTrue(np.array_equal(sample.seg_data,
                                    getattr(self, varname).seg_data))
                    if param in ["minmax", "grayscale"]:
                        if param == "minmax":
                            self.assertTrue(np.min(sample.img_data) >= 0)
                            self.assertTrue(np.max(sample.img_data) <= 1)
                        else:
                            self.assertEqual(np.min(sample.img_data), 0)
                            self.assertEqual(np.max(sample.img_data), 255)
                    else:
                        self.assertTrue(np.min(sample.img_data) <= -1.0)
                        self.assertTrue(np.max(sample.img_data) >= +1.0)

    def test_SUBFUNCTIONS_NORMALIZATION_postprocessing(self):
        modi_list = ["z-score", "minmax", "grayscale"]
        for modus in modi_list:
            # Initialize Subfunction
            sf = Normalization(mode=modus)
            # Test for 2D and 3D
            for dim in ["2D", "3D"]:
                # Create sample objects
                sample_pred = deepcopy(getattr(self, "sample" + dim))
                sample_train = deepcopy(getattr(self, "sample" + dim + "seg"))
                # Run preprocessing of the subfunction
                sf.preprocessing(sample_train, training=True)
                sf.preprocessing(sample_pred, training=False)
                # Transform segmentation data to simulate prediction data
                sample_pred.pred_data = np.squeeze(sample_train.seg_data, axis=-1)
                # Run postprocessing of the subfunction
                pred = sf.postprocessing(sample_pred, sample_pred.pred_data)
                # Check for correctness
                self.assertTrue(np.array_equal(pred, sample_pred.pred_data))


    #-------------------------------------------------#
    #                   TransformHU                   #
    #-------------------------------------------------#
    def test_SUBFUNCTIONS_TransformHU_preprocessing(self):
        # Initialize Subfunction
        sf = TransformHU()
        # Test for 2D and 3D
        for dim in ["2D", "3D"]:
            # Test for training as well as prediction
            for train in [True, False]:
                # Create sample object from template
                varname = "sample" + dim
                if train : varname += "seg"
                sample = deepcopy(getattr(self, varname))
                if not train:
                    sample.details = {"slope":1.0, "intercept":-1024.0}
                # Run preprocessing of the subfunction
                sf.preprocessing(sample, training=train)
                # Check for correctness
                self.assertTrue(np.array_equal(sample.seg_data,
                                getattr(self, varname).seg_data))
                self.assertFalse(np.array_equal(sample.img_data,
                                getattr(self, varname).img_data))

    def test_SUBFUNCTIONS_TransformHU_postprocessing(self):
        # Initialize Subfunction
        sf = TransformHU()
        # Test for 2D and 3D
        for dim in ["2D", "3D"]:
            # Create sample objects
            sample_pred = deepcopy(getattr(self, "sample" + dim))
            sample_train = deepcopy(getattr(self, "sample" + dim + "seg"))
            sample_pred.details = {"slope":1.0, "intercept":-1024.0}
            sample_train.details = {"slope":1.0, "intercept":-1024.0}
            # Run preprocessing of the subfunction
            sf.preprocessing(sample_train, training=True)
            sf.preprocessing(sample_pred, training=False)
            # Transform segmentation data to simulate prediction data
            sample_pred.pred_data = np.squeeze(sample_train.seg_data, axis=-1)
            # Run postprocessing of the subfunction
            pred = sf.postprocessing(sample_pred, sample_pred.pred_data)
            # Check for correctness
            self.assertTrue(np.array_equal(pred, sample_pred.pred_data))