diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md index 4405a4b5..f9a105a0 100644 --- a/CONTRIBUTING.md +++ b/CONTRIBUTING.md @@ -84,6 +84,11 @@ In train config file (if exists), please follow the following requirements in or We prepared several premerge CI tests to verify your bundle. +### Prepare dependencies + +For dependencies that support `pip install` command, please put them into `optional_packages_version` in `configs/metadata.json` ([click here for instance](https://github.com/Project-MONAI/model-zoo/blob/dev/models/brats_mri_segmentation/configs/metadata.json)), and the CI test program will extract and install all libraries directly before running tests. +For dependencies that require multiple steps to install, please prepare a install script in `ci/install_scripts`, and put the bundle name and the script path into `install_dependency_dict` in `ci/bundle_custom_data.py` ([click here for instance](https://github.com/Project-MONAI/model-zoo/tree/dev/ci/install_scripts/)). + ### Necessary verifications 1. Check if necessary files are existing in your bundle. diff --git a/ci/bundle_custom_data.py b/ci/bundle_custom_data.py index a0da3d7f..dbf91530 100644 --- a/ci/bundle_custom_data.py +++ b/ci/bundle_custom_data.py @@ -13,7 +13,12 @@ # This list is used for our CI tests to determine whether a bundle needs to be tested with # the `verify_data_shape` function in `verify_bundle.py`. # If a bundle does not need to be tested, please add the bundle name into the list. -exclude_verify_shape_list = ["mednist_gan", "lung_nodule_ct_detection", "pathology_nuclei_segmentation_classification"] +exclude_verify_shape_list = [ + "mednist_gan", + "lung_nodule_ct_detection", + "pathology_nuclei_segmentation_classification", + "brats_mri_generative_diffusion", +] # This list is used for our CI tests to determine whether a bundle contains the preferred files. # If a bundle does not have any of the preferred files, please add the bundle name into the list. @@ -28,8 +33,16 @@ "wholeBrainSeg_Large_UNEST_segmentation", "breast_density_classification", "mednist_reg", + "brats_mri_generative_diffusion", ] +# This dict is used for our CI tests to install required dependencies that cannot be installed by `pip install` directly. +# If a bundle has this kind of dependencies, please add the bundle name (key), and the path of the install script (value) +# into the dict. +install_dependency_dict = { + "brats_mri_generative_diffusion": "ci/install_scripts/install_brats_mri_generative_diffusion_dependency.sh" +} + # This list is used for our CI tests to determine whether a bundle supports TensorRT export. Related # test will be employed for bundles in the list. include_verify_tensorrt_list = ["spleen_ct_segmentation", "endoscopic_tool_segmentation", "pathology_tumor_detection"] diff --git a/ci/get_bundle_requirements.py b/ci/get_bundle_requirements.py index 03dddf06..e9de7a2e 100644 --- a/ci/get_bundle_requirements.py +++ b/ci/get_bundle_requirements.py @@ -13,6 +13,7 @@ import argparse import os +from bundle_custom_data import install_dependency_dict from utils import get_json_dict @@ -44,11 +45,24 @@ def get_requirements(bundle, models_path): print(requirements_file_name) +def get_install_script(bundle): + # install extra dependencies if needed + script_path = "" + if bundle in install_dependency_dict.keys(): + script_path = install_dependency_dict[bundle] + print(script_path) + + if __name__ == "__main__": parser = argparse.ArgumentParser(description="") parser.add_argument("-b", "--b", type=str, help="bundle name.") parser.add_argument("-p", "--p", type=str, default="models", help="models path.") + parser.add_argument("--get_script", type=bool, default=False, help="whether to get the install script.") args = parser.parse_args() bundle = args.b models_path = args.p - get_requirements(bundle, models_path) + get_script = args.get_script + if get_script is True: + get_install_script(bundle) + else: + get_requirements(bundle, models_path) diff --git a/ci/install_scripts/install_brats_mri_generative_diffusion_dependency.sh b/ci/install_scripts/install_brats_mri_generative_diffusion_dependency.sh new file mode 100644 index 00000000..13207a1f --- /dev/null +++ b/ci/install_scripts/install_brats_mri_generative_diffusion_dependency.sh @@ -0,0 +1,18 @@ +# Copyright (c) MONAI Consortium +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# http://www.apache.org/licenses/LICENSE-2.0 +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +# install cython to avoid error: +# https://github.com/Project-MONAI/model-zoo/actions/runs/4980530305/jobs/8913560199#step:2:550 +git clone https://github.com/Project-MONAI/GenerativeModels.git +cd GenerativeModels/ +git checkout f969c24f88d013dc0045fb7b2885a01fb219992b +python setup.py install +cd .. diff --git a/ci/run_premerge_gpu.sh b/ci/run_premerge_gpu.sh index 4a7abcc5..e3240348 100755 --- a/ci/run_premerge_gpu.sh +++ b/ci/run_premerge_gpu.sh @@ -65,6 +65,12 @@ verify_bundle() { echo "install required libraries for bundle: $bundle" pipenv install -r "$requirements" fi + # get extra install script if exists + extra_script=$(pipenv run python $(pwd)/ci/get_bundle_requirements.py --b "$bundle" --get_script True) + if [ ! -z "$extra_script" ]; then + echo "install extra libraries with script: $extra_script" + bash $extra_script + fi # verify bundle pipenv run python $(pwd)/ci/verify_bundle.py --b "$bundle" remove_pipenv diff --git a/ci/verify_bundle.py b/ci/verify_bundle.py index a337283c..b48f800b 100644 --- a/ci/verify_bundle.py +++ b/ci/verify_bundle.py @@ -47,6 +47,20 @@ def _find_bundle_file(root_dir: str, file: str, suffix=("json", "yaml", "yml")): return file_name +def _get_weights_names(bundle: str): + # TODO: this function is temporarily used. It should be replaced by detailed config tests. + if bundle == "brats_mri_generative_diffusion": + return "model_autoencoder.pt", "model_autoencoder.ts" + return "model.pt", "model.ts" + + +def _get_net_id(bundle: str): + # TODO: this function is temporarily used. It should be replaced by detailed config tests. + if bundle == "brats_mri_generative_diffusion": + return "autoencoder_def" + return "network_def" + + def _check_missing_keys(file_name: str, bundle_path: str, keys_list: List): config = ConfigParser.load_config_file(os.path.join(bundle_path, "configs", file_name)) missing_keys = [] @@ -267,7 +281,9 @@ def verify_data_shape(bundle_path: str, net_id: str, config_file: str): ) -def verify_torchscript(bundle_path: str, net_id: str, config_file: str): +def verify_torchscript( + bundle_path: str, net_id: str, config_file: str, model_name: str = "model.pt", ts_name: str = "model.ts" +): """ This function is used to verify if the checkpoint is able to export into torchscript model, and if "models/model.ts" is provided, it will be checked if it is able to be loaded @@ -277,14 +293,14 @@ def verify_torchscript(bundle_path: str, net_id: str, config_file: str): ckpt_export( net_id=net_id, filepath=os.path.join(bundle_path, "models/verify_model.ts"), - ckpt_file=os.path.join(bundle_path, "models/model.pt"), + ckpt_file=os.path.join(bundle_path, "models", model_name), meta_file=os.path.join(bundle_path, "configs/metadata.json"), config_file=os.path.join(bundle_path, config_file), bundle_root=bundle_path, ) print("export weights into TorchScript module successfully.") - ts_model_path = os.path.join(bundle_path, "models/model.ts") + ts_model_path = os.path.join(bundle_path, "models", ts_name) if os.path.exists(ts_model_path): torch.jit.load(ts_model_path) print("Provided TorchScript module is verified correctly.") @@ -313,7 +329,8 @@ def verify(bundle, models_path="models", mode="full"): return # The following are optional tests and require GPU - net_id, inference_file_name = "network_def", _find_bundle_file(os.path.join(bundle_path, "configs"), "inference") + net_id = _get_net_id(bundle) + inference_file_name = _find_bundle_file(os.path.join(bundle_path, "configs"), "inference") config_file = os.path.join("configs", inference_file_name) if bundle in exclude_verify_shape_list: @@ -325,7 +342,8 @@ def verify(bundle, models_path="models", mode="full"): if bundle in exclude_verify_torchscript_list: print(f"bundle: {bundle} does not support torchscript, skip verifying.") else: - verify_torchscript(bundle_path, net_id, config_file) + model_name, ts_name = _get_weights_names(bundle=bundle) + verify_torchscript(bundle_path, net_id, config_file, model_name, ts_name) if __name__ == "__main__": diff --git a/models/brats_mri_generative_diffusion/LICENSE b/models/brats_mri_generative_diffusion/LICENSE new file mode 100644 index 00000000..261eeb9e --- /dev/null +++ b/models/brats_mri_generative_diffusion/LICENSE @@ -0,0 +1,201 @@ + Apache License + Version 2.0, January 2004 + http://www.apache.org/licenses/ + + TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION + + 1. Definitions. + + "License" shall mean the terms and conditions for use, reproduction, + and distribution as defined by Sections 1 through 9 of this document. + + "Licensor" shall mean the copyright owner or entity authorized by + the copyright owner that is granting the License. + + "Legal Entity" shall mean the union of the acting entity and all + other entities that control, are controlled by, or are under common + control with that entity. For the purposes of this definition, + "control" means (i) the power, direct or indirect, to cause the + direction or management of such entity, whether by contract or + otherwise, or (ii) ownership of fifty percent (50%) or more of the + outstanding shares, or (iii) beneficial ownership of such entity. + + "You" (or "Your") shall mean an individual or Legal Entity + exercising permissions granted by this License. + + "Source" form shall mean the preferred form for making modifications, + including but not limited to software source code, documentation + source, and configuration files. + + "Object" form shall mean any form resulting from mechanical + transformation or translation of a Source form, including but + not limited to compiled object code, generated documentation, + and conversions to other media types. + + "Work" shall mean the work of authorship, whether in Source or + Object form, made available under the License, as indicated by a + copyright notice that is included in or attached to the work + (an example is provided in the Appendix below). + + "Derivative Works" shall mean any work, whether in Source or Object + form, that is based on (or derived from) the Work and for which the + editorial revisions, annotations, elaborations, or other modifications + represent, as a whole, an original work of authorship. For the purposes + of this License, Derivative Works shall not include works that remain + separable from, or merely link (or bind by name) to the interfaces of, + the Work and Derivative Works thereof. + + "Contribution" shall mean any work of authorship, including + the original version of the Work and any modifications or additions + to that Work or Derivative Works thereof, that is intentionally + submitted to Licensor for inclusion in the Work by the copyright owner + or by an individual or Legal Entity authorized to submit on behalf of + the copyright owner. For the purposes of this definition, "submitted" + means any form of electronic, verbal, or written communication sent + to the Licensor or its representatives, including but not limited to + communication on electronic mailing lists, source code control systems, + and issue tracking systems that are managed by, or on behalf of, the + Licensor for the purpose of discussing and improving the Work, but + excluding communication that is conspicuously marked or otherwise + designated in writing by the copyright owner as "Not a Contribution." + + "Contributor" shall mean Licensor and any individual or Legal Entity + on behalf of whom a Contribution has been received by Licensor and + subsequently incorporated within the Work. + + 2. Grant of Copyright License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + copyright license to reproduce, prepare Derivative Works of, + publicly display, publicly perform, sublicense, and distribute the + Work and such Derivative Works in Source or Object form. + + 3. Grant of Patent License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + (except as stated in this section) patent license to make, have made, + use, offer to sell, sell, import, and otherwise transfer the Work, + where such license applies only to those patent claims licensable + by such Contributor that are necessarily infringed by their + Contribution(s) alone or by combination of their Contribution(s) + with the Work to which such Contribution(s) was submitted. If You + institute patent litigation against any entity (including a + cross-claim or counterclaim in a lawsuit) alleging that the Work + or a Contribution incorporated within the Work constitutes direct + or contributory patent infringement, then any patent licenses + granted to You under this License for that Work shall terminate + as of the date such litigation is filed. + + 4. Redistribution. You may reproduce and distribute copies of the + Work or Derivative Works thereof in any medium, with or without + modifications, and in Source or Object form, provided that You + meet the following conditions: + + (a) You must give any other recipients of the Work or + Derivative Works a copy of this License; and + + (b) You must cause any modified files to carry prominent notices + stating that You changed the files; and + + (c) You must retain, in the Source form of any Derivative Works + that You distribute, all copyright, patent, trademark, and + attribution notices from the Source form of the Work, + excluding those notices that do not pertain to any part of + the Derivative Works; and + + (d) If the Work includes a "NOTICE" text file as part of its + distribution, then any Derivative Works that You distribute must + include a readable copy of the attribution notices contained + within such NOTICE file, excluding those notices that do not + pertain to any part of the Derivative Works, in at least one + of the following places: within a NOTICE text file distributed + as part of the Derivative Works; within the Source form or + documentation, if provided along with the Derivative Works; or, + within a display generated by the Derivative Works, if and + wherever such third-party notices normally appear. The contents + of the NOTICE file are for informational purposes only and + do not modify the License. You may add Your own attribution + notices within Derivative Works that You distribute, alongside + or as an addendum to the NOTICE text from the Work, provided + that such additional attribution notices cannot be construed + as modifying the License. + + You may add Your own copyright statement to Your modifications and + may provide additional or different license terms and conditions + for use, reproduction, or distribution of Your modifications, or + for any such Derivative Works as a whole, provided Your use, + reproduction, and distribution of the Work otherwise complies with + the conditions stated in this License. + + 5. Submission of Contributions. Unless You explicitly state otherwise, + any Contribution intentionally submitted for inclusion in the Work + by You to the Licensor shall be under the terms and conditions of + this License, without any additional terms or conditions. + Notwithstanding the above, nothing herein shall supersede or modify + the terms of any separate license agreement you may have executed + with Licensor regarding such Contributions. + + 6. Trademarks. This License does not grant permission to use the trade + names, trademarks, service marks, or product names of the Licensor, + except as required for reasonable and customary use in describing the + origin of the Work and reproducing the content of the NOTICE file. + + 7. Disclaimer of Warranty. Unless required by applicable law or + agreed to in writing, Licensor provides the Work (and each + Contributor provides its Contributions) on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or + implied, including, without limitation, any warranties or conditions + of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A + PARTICULAR PURPOSE. You are solely responsible for determining the + appropriateness of using or redistributing the Work and assume any + risks associated with Your exercise of permissions under this License. + + 8. Limitation of Liability. In no event and under no legal theory, + whether in tort (including negligence), contract, or otherwise, + unless required by applicable law (such as deliberate and grossly + negligent acts) or agreed to in writing, shall any Contributor be + liable to You for damages, including any direct, indirect, special, + incidental, or consequential damages of any character arising as a + result of this License or out of the use or inability to use the + Work (including but not limited to damages for loss of goodwill, + work stoppage, computer failure or malfunction, or any and all + other commercial damages or losses), even if such Contributor + has been advised of the possibility of such damages. + + 9. Accepting Warranty or Additional Liability. While redistributing + the Work or Derivative Works thereof, You may choose to offer, + and charge a fee for, acceptance of support, warranty, indemnity, + or other liability obligations and/or rights consistent with this + License. However, in accepting such obligations, You may act only + on Your own behalf and on Your sole responsibility, not on behalf + of any other Contributor, and only if You agree to indemnify, + defend, and hold each Contributor harmless for any liability + incurred by, or claims asserted against, such Contributor by reason + of your accepting any such warranty or additional liability. + + END OF TERMS AND CONDITIONS + + APPENDIX: How to apply the Apache License to your work. + + To apply the Apache License to your work, attach the following + boilerplate notice, with the fields enclosed by brackets "[]" + replaced with your own identifying information. (Don't include + the brackets!) The text should be enclosed in the appropriate + comment syntax for the file format. We also recommend that a + file or class name and description of purpose be included on the + same "printed page" as the copyright notice for easier + identification within third-party archives. + + Copyright [yyyy] [name of copyright owner] + + Licensed under the Apache License, Version 2.0 (the "License"); + you may not use this file except in compliance with the License. + You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + + Unless required by applicable law or agreed to in writing, software + distributed under the License is distributed on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + See the License for the specific language governing permissions and + limitations under the License. diff --git a/models/brats_mri_generative_diffusion/configs/inference.json b/models/brats_mri_generative_diffusion/configs/inference.json new file mode 100644 index 00000000..b2965eea --- /dev/null +++ b/models/brats_mri_generative_diffusion/configs/inference.json @@ -0,0 +1,101 @@ +{ + "imports": [ + "$import torch", + "$from datetime import datetime", + "$from pathlib import Path" + ], + "bundle_root": ".", + "model_dir": "$@bundle_root + '/models'", + "output_dir": "$@bundle_root + '/output'", + "create_output_dir": "$Path(@output_dir).mkdir(exist_ok=True)", + "device": "$torch.device('cuda:0' if torch.cuda.is_available() else 'cpu')", + "output_postfix": "$datetime.now().strftime('sample_%Y%m%d_%H%M%S')", + "spatial_dims": 3, + "image_channels": 1, + "latent_channels": 8, + "latent_shape": [ + 8, + 36, + 44, + 28 + ], + "autoencoder_def": { + "_target_": "generative.networks.nets.AutoencoderKL", + "spatial_dims": "@spatial_dims", + "in_channels": "@image_channels", + "out_channels": "@image_channels", + "latent_channels": "@latent_channels", + "num_channels": [ + 64, + 128, + 256 + ], + "num_res_blocks": 2, + "norm_num_groups": 32, + "norm_eps": 1e-06, + "attention_levels": [ + false, + false, + false + ], + "with_encoder_nonlocal_attn": false, + "with_decoder_nonlocal_attn": false + }, + "network_def": { + "_target_": "generative.networks.nets.DiffusionModelUNet", + "spatial_dims": "@spatial_dims", + "in_channels": "@latent_channels", + "out_channels": "@latent_channels", + "num_channels": [ + 256, + 256, + 512 + ], + "attention_levels": [ + false, + true, + true + ], + "num_head_channels": [ + 0, + 64, + 64 + ], + "num_res_blocks": 2 + }, + "load_autoencoder_path": "$@bundle_root + '/models/model_autoencoder.pt'", + "load_autoencoder": "$@autoencoder_def.load_state_dict(torch.load(@load_autoencoder_path))", + "autoencoder": "$@autoencoder_def.to(@device)", + "load_diffusion_path": "$@model_dir + '/model.pt'", + "load_diffusion": "$@network_def.load_state_dict(torch.load(@load_diffusion_path))", + "diffusion": "$@network_def.to(@device)", + "noise_scheduler": { + "_target_": "generative.networks.schedulers.DDIMScheduler", + "_requires_": [ + "@load_diffusion", + "@load_autoencoder" + ], + "num_train_timesteps": 1000, + "beta_start": 0.0015, + "beta_end": 0.0195, + "beta_schedule": "scaled_linear", + "clip_sample": false + }, + "noise": "$torch.randn([1]+@latent_shape).to(@device)", + "set_timesteps": "$@noise_scheduler.set_timesteps(num_inference_steps=50)", + "inferer": { + "_target_": "scripts.ldm_sampler.LDMSampler", + "_requires_": "@set_timesteps" + }, + "sample": "$@inferer.sampling_fn(@noise, @autoencoder, @diffusion, @noise_scheduler)", + "saver": { + "_target_": "SaveImage", + "_requires_": "@create_output_dir", + "output_dir": "@output_dir", + "output_postfix": "@output_postfix" + }, + "generated_image": "$@sample", + "run": [ + "$@saver(@generated_image[0])" + ] +} diff --git a/models/brats_mri_generative_diffusion/configs/inference_autoencoder.json b/models/brats_mri_generative_diffusion/configs/inference_autoencoder.json new file mode 100644 index 00000000..501f342b --- /dev/null +++ b/models/brats_mri_generative_diffusion/configs/inference_autoencoder.json @@ -0,0 +1,149 @@ +{ + "imports": [ + "$import torch", + "$from datetime import datetime", + "$from pathlib import Path" + ], + "bundle_root": ".", + "model_dir": "$@bundle_root + '/models'", + "dataset_dir": "@bundle_root", + "output_dir": "$@bundle_root + '/output'", + "create_output_dir": "$Path(@output_dir).mkdir(exist_ok=True)", + "device": "$torch.device('cuda:0' if torch.cuda.is_available() else 'cpu')", + "output_orig_postfix": "recon", + "output_recon_postfix": "orig", + "channel": 0, + "spacing": [ + 1.1, + 1.1, + 1.1 + ], + "spatial_dims": 3, + "image_channels": 1, + "latent_channels": 8, + "infer_patch_size": [ + 144, + 176, + 112 + ], + "autoencoder_def": { + "_target_": "generative.networks.nets.AutoencoderKL", + "spatial_dims": "@spatial_dims", + "in_channels": "@image_channels", + "out_channels": "@image_channels", + "latent_channels": "@latent_channels", + "num_channels": [ + 64, + 128, + 256 + ], + "num_res_blocks": 2, + "norm_num_groups": 32, + "norm_eps": 1e-06, + "attention_levels": [ + false, + false, + false + ], + "with_encoder_nonlocal_attn": false, + "with_decoder_nonlocal_attn": false + }, + "load_autoencoder_path": "$@bundle_root + '/models/model_autoencoder.pt'", + "load_autoencoder": "$@autoencoder_def.load_state_dict(torch.load(@load_autoencoder_path))", + "autoencoder": "$@autoencoder_def.to(@device)", + "preprocessing_transforms": [ + { + "_target_": "LoadImaged", + "keys": "image" + }, + { + "_target_": "EnsureChannelFirstd", + "keys": "image" + }, + { + "_target_": "Lambdad", + "keys": "image", + "func": "$lambda x: x[@channel, :, :, :]" + }, + { + "_target_": "AddChanneld", + "keys": "image" + }, + { + "_target_": "EnsureTyped", + "keys": "image" + }, + { + "_target_": "Orientationd", + "keys": "image", + "axcodes": "RAS" + }, + { + "_target_": "Spacingd", + "keys": "image", + "pixdim": "@spacing", + "mode": "bilinear" + } + ], + "crop_transforms": [ + { + "_target_": "CenterSpatialCropd", + "keys": "image", + "roi_size": "@infer_patch_size" + } + ], + "final_transforms": [ + { + "_target_": "ScaleIntensityRangePercentilesd", + "keys": "image", + "lower": 0, + "upper": 99.5, + "b_min": 0, + "b_max": 1 + } + ], + "preprocessing": { + "_target_": "Compose", + "transforms": "$@preprocessing_transforms + @crop_transforms + @final_transforms" + }, + "dataset": { + "_target_": "monai.apps.DecathlonDataset", + "root_dir": "@dataset_dir", + "task": "Task01_BrainTumour", + "section": "validation", + "cache_rate": 0.0, + "num_workers": 8, + "download": false, + "transform": "@preprocessing" + }, + "dataloader": { + "_target_": "DataLoader", + "dataset": "@dataset", + "batch_size": 1, + "shuffle": true, + "num_workers": 0 + }, + "saver_orig": { + "_target_": "SaveImage", + "_requires_": "@create_output_dir", + "output_dir": "@output_dir", + "output_postfix": "@output_orig_postfix", + "resample": false, + "padding_mode": "zeros" + }, + "saver_recon": { + "_target_": "SaveImage", + "_requires_": "@create_output_dir", + "output_dir": "@output_dir", + "output_postfix": "@output_recon_postfix", + "resample": false, + "padding_mode": "zeros" + }, + "input_img": "$monai.utils.first(@dataloader)['image'].to(@device)", + "recon_img": "$@autoencoder(@input_img)[0][0]", + "run": [ + "$@load_autoencoder", + "$@saver_orig(@input_img[0][0])", + "$@saver_recon(@recon_img)" + ] +} diff --git a/models/brats_mri_generative_diffusion/configs/logging.conf b/models/brats_mri_generative_diffusion/configs/logging.conf new file mode 100644 index 00000000..91c1a21c --- /dev/null +++ b/models/brats_mri_generative_diffusion/configs/logging.conf @@ -0,0 +1,21 @@ +[loggers] +keys=root + +[handlers] +keys=consoleHandler + +[formatters] +keys=fullFormatter + +[logger_root] +level=INFO +handlers=consoleHandler + +[handler_consoleHandler] +class=StreamHandler +level=INFO +formatter=fullFormatter +args=(sys.stdout,) + +[formatter_fullFormatter] +format=%(asctime)s - %(name)s - %(levelname)s - %(message)s diff --git a/models/brats_mri_generative_diffusion/configs/metadata.json b/models/brats_mri_generative_diffusion/configs/metadata.json new file mode 100644 index 00000000..4bcc437d --- /dev/null +++ b/models/brats_mri_generative_diffusion/configs/metadata.json @@ -0,0 +1,119 @@ +{ + "schema": "https://github.com/Project-MONAI/MONAI-extra-test-data/releases/download/0.8.1/meta_schema_generator_ldm_20230507.json", + "version": "1.0.0", + "changelog": { + "1.0.0": "Initial release" + }, + "monai_version": "1.2.0rc5", + "pytorch_version": "1.13.1", + "numpy_version": "1.22.2", + "optional_packages_version": { + "nibabel": "5.1.0", + "lpips": "0.1.4" + }, + "name": "BraTS MRI image latent diffusion generation", + "task": "BraTS MRI image synthesis", + "description": "A generative model for creating 3D brain MRI from Gaussian noise based on BraTS dataset", + "authors": "MONAI team", + "copyright": "Copyright (c) MONAI Consortium", + "data_source": "http://medicaldecathlon.com/", + "data_type": "nibabel", + "image_classes": "Flair brain MRI with 1.1x1.1x1.1 mm voxel size", + "eval_metrics": {}, + "intended_use": "This is a research tool/prototype and not to be used clinically", + "references": [], + "autoencoder_data_format": { + "inputs": { + "image": { + "type": "image", + "format": "image", + "num_channels": 1, + "spatial_shape": [ + 112, + 128, + 80 + ], + "dtype": "float32", + "value_range": [ + 0, + 1 + ], + "is_patch_data": true + } + }, + "outputs": { + "pred": { + "type": "image", + "format": "image", + "num_channels": 1, + "spatial_shape": [ + 112, + 128, + 80 + ], + "dtype": "float32", + "value_range": [ + 0, + 1 + ], + "is_patch_data": true, + "channel_def": { + "0": "image" + } + } + } + }, + "generator_data_format": { + "inputs": { + "latent": { + "type": "noise", + "format": "image", + "num_channels": 8, + "spatial_shape": [ + 36, + 44, + 28 + ], + "dtype": "float32", + "value_range": [ + 0, + 1 + ], + "is_patch_data": true + }, + "condition": { + "type": "timesteps", + "format": "timesteps", + "num_channels": 1, + "spatial_shape": [], + "dtype": "long", + "value_range": [ + 0, + 1000 + ], + "is_patch_data": false + } + }, + "outputs": { + "pred": { + "type": "feature", + "format": "image", + "num_channels": 8, + "spatial_shape": [ + 36, + 44, + 28 + ], + "dtype": "float32", + "value_range": [ + 0, + 1 + ], + "is_patch_data": true, + "channel_def": { + "0": "image" + } + } + } + } +} diff --git a/models/brats_mri_generative_diffusion/configs/multi_gpu_train_autoencoder.json b/models/brats_mri_generative_diffusion/configs/multi_gpu_train_autoencoder.json new file mode 100644 index 00000000..e0061ff4 --- /dev/null +++ b/models/brats_mri_generative_diffusion/configs/multi_gpu_train_autoencoder.json @@ -0,0 +1,40 @@ +{ + "device": "$torch.device(f'cuda:{dist.get_rank()}')", + "gnetwork": { + "_target_": "torch.nn.parallel.DistributedDataParallel", + "module": "$@autoencoder_def.to(@device)", + "device_ids": [ + "@device" + ] + }, + "dnetwork": { + "_target_": "torch.nn.parallel.DistributedDataParallel", + "module": "$@discriminator_def.to(@device)", + "device_ids": [ + "@device" + ] + }, + "train#sampler": { + "_target_": "DistributedSampler", + "dataset": "@train#dataset", + "even_divisible": true, + "shuffle": true + }, + "train#dataloader#sampler": "@train#sampler", + "train#dataloader#shuffle": false, + "train#trainer#train_handlers": "$@train#handlers[: -2 if dist.get_rank() > 0 else None]", + "initialize": [ + "$import torch.distributed as dist", + "$dist.is_initialized() or dist.init_process_group(backend='nccl')", + "$torch.cuda.set_device(@device)", + "$monai.utils.set_determinism(seed=123)", + "$import logging", + "$@train#trainer.logger.setLevel(logging.WARNING if dist.get_rank() > 0 else logging.INFO)" + ], + "run": [ + "$@train#trainer.run()" + ], + "finalize": [ + "$dist.is_initialized() and dist.destroy_process_group()" + ] +} diff --git a/models/brats_mri_generative_diffusion/configs/multi_gpu_train_diffusion.json b/models/brats_mri_generative_diffusion/configs/multi_gpu_train_diffusion.json new file mode 100644 index 00000000..fbf130d2 --- /dev/null +++ b/models/brats_mri_generative_diffusion/configs/multi_gpu_train_diffusion.json @@ -0,0 +1,16 @@ +{ + "diffusion": { + "_target_": "torch.nn.parallel.DistributedDataParallel", + "module": "$@network_def.to(@device)", + "device_ids": [ + "@device" + ], + "find_unused_parameters": true + }, + "run": [ + "@load_autoencoder", + "$@autoencoder.eval()", + "$print('scale factor:',@scale_factor)", + "$@train#trainer.run()" + ] +} diff --git a/models/brats_mri_generative_diffusion/configs/train_autoencoder.json b/models/brats_mri_generative_diffusion/configs/train_autoencoder.json new file mode 100644 index 00000000..28136f51 --- /dev/null +++ b/models/brats_mri_generative_diffusion/configs/train_autoencoder.json @@ -0,0 +1,206 @@ +{ + "imports": [ + "$import functools", + "$import glob", + "$import scripts" + ], + "bundle_root": ".", + "device": "$torch.device('cuda:0' if torch.cuda.is_available() else 'cpu')", + "ckpt_dir": "$@bundle_root + '/models'", + "tf_dir": "$@bundle_root + '/eval'", + "dataset_dir": "/workspace/data/medical", + "pretrained": false, + "perceptual_loss_model_weights_path": null, + "train_batch_size": 2, + "lr": 1e-05, + "train_patch_size": [ + 112, + 128, + 80 + ], + "channel": 0, + "spacing": [ + 1.1, + 1.1, + 1.1 + ], + "spatial_dims": 3, + "image_channels": 1, + "latent_channels": 8, + "discriminator_def": { + "_target_": "generative.networks.nets.PatchDiscriminator", + "spatial_dims": "@spatial_dims", + "num_layers_d": 3, + "num_channels": 32, + "in_channels": 1, + "out_channels": 1, + "norm": "INSTANCE" + }, + "autoencoder_def": { + "_target_": "generative.networks.nets.AutoencoderKL", + "spatial_dims": "@spatial_dims", + "in_channels": "@image_channels", + "out_channels": "@image_channels", + "latent_channels": "@latent_channels", + "num_channels": [ + 64, + 128, + 256 + ], + "num_res_blocks": 2, + "norm_num_groups": 32, + "norm_eps": 1e-06, + "attention_levels": [ + false, + false, + false + ], + "with_encoder_nonlocal_attn": false, + "with_decoder_nonlocal_attn": false + }, + "perceptual_loss_def": { + "_target_": "generative.losses.PerceptualLoss", + "spatial_dims": "@spatial_dims", + "network_type": "resnet50", + "is_fake_3d": true, + "fake_3d_ratio": 0.2, + "pretrained": "@pretrained", + "pretrained_path": "@perceptual_loss_model_weights_path", + "pretrained_state_dict_key": "state_dict" + }, + "dnetwork": "$@discriminator_def.to(@device)", + "gnetwork": "$@autoencoder_def.to(@device)", + "loss_perceptual": "$@perceptual_loss_def.to(@device)", + "doptimizer": { + "_target_": "torch.optim.Adam", + "params": "$@dnetwork.parameters()", + "lr": "@lr" + }, + "goptimizer": { + "_target_": "torch.optim.Adam", + "params": "$@gnetwork.parameters()", + "lr": "@lr" + }, + "preprocessing_transforms": [ + { + "_target_": "LoadImaged", + "keys": "image" + }, + { + "_target_": "EnsureChannelFirstd", + "keys": "image" + }, + { + "_target_": "Lambdad", + "keys": "image", + "func": "$lambda x: x[@channel, :, :, :]" + }, + { + "_target_": "AddChanneld", + "keys": "image" + }, + { + "_target_": "EnsureTyped", + "keys": "image" + }, + { + "_target_": "Orientationd", + "keys": "image", + "axcodes": "RAS" + }, + { + "_target_": "Spacingd", + "keys": "image", + "pixdim": "@spacing", + "mode": "bilinear" + } + ], + "final_transforms": [ + { + "_target_": "ScaleIntensityRangePercentilesd", + "keys": "image", + "lower": 0, + "upper": 99.5, + "b_min": 0, + "b_max": 1 + } + ], + "train": { + "crop_transforms": [ + { + "_target_": "RandSpatialCropd", + "keys": "image", + "roi_size": "@train_patch_size", + "random_size": false + } + ], + "preprocessing": { + "_target_": "Compose", + "transforms": "$@preprocessing_transforms + @train#crop_transforms + @final_transforms" + }, + "dataset": { + "_target_": "monai.apps.DecathlonDataset", + "root_dir": "@dataset_dir", + "task": "Task01_BrainTumour", + "section": "training", + "cache_rate": 1.0, + "num_workers": 8, + "download": false, + "transform": "@train#preprocessing" + }, + "dataloader": { + "_target_": "DataLoader", + "dataset": "@train#dataset", + "batch_size": "@train_batch_size", + "shuffle": true, + "num_workers": 0 + }, + "handlers": [ + { + "_target_": "CheckpointSaver", + "save_dir": "@ckpt_dir", + "save_dict": { + "model": "@gnetwork" + }, + "save_interval": 0, + "save_final": true, + "epoch_level": true, + "final_filename": "model_autoencoder.pt" + }, + { + "_target_": "StatsHandler", + "tag_name": "train_loss", + "output_transform": "$lambda x: monai.handlers.from_engine(['g_loss'], first=True)(x)[0]" + }, + { + "_target_": "TensorBoardStatsHandler", + "log_dir": "@tf_dir", + "tag_name": "train_loss", + "output_transform": "$lambda x: monai.handlers.from_engine(['g_loss'], first=True)(x)[0]" + } + ], + "trainer": { + "_target_": "scripts.ldm_trainer.VaeGanTrainer", + "device": "@device", + "max_epochs": 1500, + "train_data_loader": "@train#dataloader", + "g_network": "@gnetwork", + "g_optimizer": "@goptimizer", + "g_loss_function": "$functools.partial(scripts.losses.generator_loss, disc_net=@dnetwork, loss_perceptual=@loss_perceptual)", + "d_network": "@dnetwork", + "d_optimizer": "@doptimizer", + "d_loss_function": "$functools.partial(scripts.losses.discriminator_loss, disc_net=@dnetwork)", + "d_train_steps": 5, + "g_update_latents": true, + "latent_shape": "@latent_channels", + "key_train_metric": "$None", + "train_handlers": "@train#handlers" + } + }, + "initialize": [ + "$monai.utils.set_determinism(seed=0)" + ], + "run": [ + "$@train#trainer.run()" + ] +} diff --git a/models/brats_mri_generative_diffusion/configs/train_diffusion.json b/models/brats_mri_generative_diffusion/configs/train_diffusion.json new file mode 100644 index 00000000..58ee2648 --- /dev/null +++ b/models/brats_mri_generative_diffusion/configs/train_diffusion.json @@ -0,0 +1,157 @@ +{ + "ckpt_dir": "$@bundle_root + '/models'", + "train_batch_size": 4, + "lr": 1e-05, + "train_patch_size": [ + 144, + 176, + 112 + ], + "latent_shape": [ + "@latent_channels", + 36, + 44, + 28 + ], + "load_autoencoder_path": "$@bundle_root + '/models/model_autoencoder.pt'", + "load_autoencoder": "$@autoencoder_def.load_state_dict(torch.load(@load_autoencoder_path))", + "autoencoder": "$@autoencoder_def.to(@device)", + "network_def": { + "_target_": "generative.networks.nets.DiffusionModelUNet", + "spatial_dims": "@spatial_dims", + "in_channels": "@latent_channels", + "out_channels": "@latent_channels", + "num_channels": [ + 256, + 256, + 512 + ], + "attention_levels": [ + false, + true, + true + ], + "num_head_channels": [ + 0, + 64, + 64 + ], + "num_res_blocks": 2 + }, + "diffusion": "$@network_def.to(@device)", + "optimizer": { + "_target_": "torch.optim.Adam", + "params": "$@diffusion.parameters()", + "lr": "@lr" + }, + "lr_scheduler": { + "_target_": "torch.optim.lr_scheduler.MultiStepLR", + "optimizer": "@optimizer", + "milestones": [ + 100, + 1000 + ], + "gamma": 0.1 + }, + "scale_factor": "$scripts.utils.compute_scale_factor(@autoencoder,@train#dataloader,@device)", + "noise_scheduler": { + "_target_": "generative.networks.schedulers.DDPMScheduler", + "_requires_": [ + "@load_autoencoder" + ], + "beta_schedule": "scaled_linear", + "num_train_timesteps": 1000, + "beta_start": 0.0015, + "beta_end": 0.0195 + }, + "inferer": { + "_target_": "generative.inferers.LatentDiffusionInferer", + "scheduler": "@noise_scheduler", + "scale_factor": "@scale_factor" + }, + "loss": { + "_target_": "torch.nn.MSELoss" + }, + "train": { + "crop_transforms": [ + { + "_target_": "CenterSpatialCropd", + "keys": "image", + "roi_size": "@train_patch_size" + } + ], + "preprocessing": { + "_target_": "Compose", + "transforms": "$@preprocessing_transforms + @train#crop_transforms + @final_transforms" + }, + "dataset": { + "_target_": "monai.apps.DecathlonDataset", + "root_dir": "@dataset_dir", + "task": "Task01_BrainTumour", + "section": "training", + "cache_rate": 1.0, + "num_workers": 8, + "download": false, + "transform": "@train#preprocessing" + }, + "dataloader": { + "_target_": "DataLoader", + "dataset": "@train#dataset", + "batch_size": "@train_batch_size", + "shuffle": true, + "num_workers": 0 + }, + "handlers": [ + { + "_target_": "LrScheduleHandler", + "lr_scheduler": "@lr_scheduler", + "print_lr": true + }, + { + "_target_": "CheckpointSaver", + "save_dir": "@ckpt_dir", + "save_dict": { + "model": "@diffusion" + }, + "save_interval": 0, + "save_final": true, + "epoch_level": true, + "final_filename": "model.pt" + }, + { + "_target_": "StatsHandler", + "tag_name": "train_diffusion_loss", + "output_transform": "$lambda x: monai.handlers.from_engine(['loss'], first=True)(x)" + }, + { + "_target_": "TensorBoardStatsHandler", + "log_dir": "@tf_dir", + "tag_name": "train_diffusion_loss", + "output_transform": "$lambda x: monai.handlers.from_engine(['loss'], first=True)(x)" + } + ], + "trainer": { + "_target_": "scripts.ldm_trainer.LDMTrainer", + "device": "@device", + "max_epochs": 5000, + "train_data_loader": "@train#dataloader", + "network": "@diffusion", + "autoencoder_model": "@autoencoder", + "optimizer": "@optimizer", + "loss_function": "@loss", + "latent_shape": "@latent_shape", + "inferer": "@inferer", + "key_train_metric": "$None", + "train_handlers": "@train#handlers" + } + }, + "initialize": [ + "$monai.utils.set_determinism(seed=0)" + ], + "run": [ + "@load_autoencoder", + "$@autoencoder.eval()", + "$print('scale factor:',@scale_factor)", + "$@train#trainer.run()" + ] +} diff --git a/models/brats_mri_generative_diffusion/docs/README.md b/models/brats_mri_generative_diffusion/docs/README.md new file mode 100644 index 00000000..f081ca3b --- /dev/null +++ b/models/brats_mri_generative_diffusion/docs/README.md @@ -0,0 +1,181 @@ +# Model Overview +A pre-trained model for volumetric (3D) Brats MRI 3D Latent Diffusion Generative Model. + +This model is trained on BraTS 2016 and 2017 data from [Medical Decathlon](http://medicaldecathlon.com/), using the Latent diffusion model [1]. + +![model workflow](https://developer.download.nvidia.com/assets/Clara/Images/monai_brain_image_gen_ldm3d_network.png) + +This model is a generator for creating images like the Flair MRIs based on BraTS 2016 and 2017 data. It was trained as a 3d latent diffusion model and accepts Gaussian random noise as inputs to produce an image output. The `train_autoencoder.json` file describes the training process of the variational autoencoder with GAN loss. The `train_diffusion.json` file describes the training process of the 3D latent diffusion model. + +In this bundle, the autoencoder uses perceptual loss, which is based on ResNet50 with pre-trained weights (the network is frozen and will not be trained in the bundle). In default, the `pretrained` parameter is specified as `False` in `train_autoencoder.json`. To ensure correct training, changing the default settings is necessary. There are two ways to utilize pretrained weights: +1. if set `pretrained` to `True`, ImageNet pretrained weights from [torchvision](https://pytorch.org/vision/stable/_modules/torchvision/models/resnet.html#ResNet50_Weights) will be used. However, the weights are for non-commercial use only. +2. if set `pretrained` to `True` and specifies the `perceptual_loss_model_weights_path` parameter, users are able to load weights from a local path. This is the way this bundle used to train, and the pre-trained weights are from some internal data. + +Please note that each user is responsible for checking the data source of the pre-trained models, the applicable licenses, and determining if suitable for the intended use. + +#### Example synthetic image +An example result from inference is shown below: +![Example synthetic image](https://developer.download.nvidia.com/assets/Clara/Images/monai_brain_image_gen_ldm3d_example_generation.png) + +**This is a demonstration network meant to just show the training process for this sort of network with MONAI. To achieve better performance, users need to use larger dataset like [Brats 2021](https://www.synapse.org/#!Synapse:syn25829067/wiki/610865) and have GPU with memory larger than 32G to enable larger networks and attention layers.** + +## MONAI Generative Model Dependencies +[MONAI generative models](https://github.com/Project-MONAI/GenerativeModels) can be installed by +``` +pip install lpips==0.1.4 +git clone https://github.com/Project-MONAI/GenerativeModels.git +cd GenerativeModels/ +git checkout f969c24f88d013dc0045fb7b2885a01fb219992b +python setup.py install +cd .. +``` + +## Data +The training data is BraTS 2016 and 2017 from the Medical Segmentation Decathalon. Users can find more details on the dataset (`Task01_BrainTumour`) at http://medicaldecathlon.com/. + +- Target: Image Generation +- Task: Synthesis +- Modality: MRI +- Size: 388 3D volumes (1 channel used) + +## Training Configuration +If you have a GPU with less than 32G of memory, you may need to decrease the batch size when training. To do so, modify the `train_batch_size` parameter in the [configs/train_autoencoder.json](../configs/train_autoencoder.json) and [configs/train_diffusion.json](../configs/train_diffusion.json) configuration files. + +### Training Configuration of Autoencoder +The autoencoder was trained using the following configuration: + +- GPU: at least 32GB GPU memory +- Actual Model Input: 112 x 128 x 80 +- AMP: False +- Optimizer: Adam +- Learning Rate: 1e-5 +- Loss: L1 loss, perceptual loss, KL divergence loss, adversarial loss, GAN BCE loss + +#### Input +1 channel 3D MRI Flair patches + +#### Output +- 1 channel 3D MRI reconstructed patches +- 8 channel mean of latent features +- 8 channel standard deviation of latent features + +### Training Configuration of Diffusion Model +The latent diffusion model was trained using the following configuration: + +- GPU: at least 32GB GPU memory +- Actual Model Input: 36 x 44 x 28 +- AMP: False +- Optimizer: Adam +- Learning Rate: 1e-5 +- Loss: MSE loss + +#### Training Input +- 8 channel noisy latent features +- an int that indicates the time step + +#### Training Output +8 channel predicted added noise + +#### Inference Input +8 channel noise + +#### Inference Output +8 channel denoised latent features + +### Memory Consumption Warning + +If you face memory issues with data loading, you can lower the caching rate `cache_rate` in the configurations within range [0, 1] to minimize the System RAM requirements. + +## Performance + +#### Training Loss +![A graph showing the autoencoder training curve](https://developer.download.nvidia.com/assets/Clara/Images/monai_brain_image_gen_ldm3d_train_autoencoder_loss.png) + +![A graph showing the latent diffusion training curve](https://developer.download.nvidia.com/assets/Clara/Images/monai_brain_image_gen_ldm3d_train_diffusion_loss.png) + +## MONAI Bundle Commands + +In addition to the Pythonic APIs, a few command line interfaces (CLI) are provided to interact with the bundle. The CLI supports flexible use cases, such as overriding configs at runtime and predefining arguments in a file. + +For more details usage instructions, visit the [MONAI Bundle Configuration Page](https://docs.monai.io/en/latest/config_syntax.html). + +### Execute Autoencoder Training + +#### Execute Autoencoder Training on single GPU + +``` +python -m monai.bundle run --config_file configs/train_autoencoder.json +``` + +Please note that if the default dataset path is not modified with the actual path (it should be the path that contains `Task01_BrainTumour`) in the bundle config files, you can also override it by using `--dataset_dir`: + +``` +python -m monai.bundle run --config_file configs/train_autoencoder.json --dataset_dir +``` + +#### Override the `train` config to execute multi-GPU training for Autoencoder +To train with multiple GPUs, use the following command, which requires scaling up the learning rate according to the number of GPUs. + +``` +torchrun --standalone --nnodes=1 --nproc_per_node=8 -m monai.bundle run --config_file "['configs/train_autoencoder.json','configs/multi_gpu_train_autoencoder.json']" --lr 8e-5 +``` + +#### Check the Autoencoder Training result +The following code generates a reconstructed image from a random input image. +We can visualize it to see if the autoencoder is trained correctly. +``` +python -m monai.bundle run --config_file configs/inference_autoencoder.json +``` + +An example of reconstructed image from inference is shown below. If the autoencoder is trained correctly, the reconstructed image should look similar to original image. + +![Example reconstructed image](https://developer.download.nvidia.com/assets/Clara/Images/monai_brain_image_gen_ldm3d_recon_example.jpg) + + +### Execute Latent Diffusion Training + +#### Execute Latent Diffusion Model Training on single GPU +After training the autoencoder, run the following command to train the latent diffusion model. This command will print out the scale factor of the latent feature space. If your autoencoder is well trained, this value should be close to 1.0. + +``` +python -m monai.bundle run --config_file "['configs/train_autoencoder.json','configs/train_diffusion.json']" +``` + +#### Override the `train` config to execute multi-GPU training for Latent Diffusion Model +To train with multiple GPUs, use the following command, which requires scaling up the learning rate according to the number of GPUs. + +``` +torchrun --standalone --nnodes=1 --nproc_per_node=8 -m monai.bundle run --config_file "['configs/train_autoencoder.json','configs/train_diffusion.json','configs/multi_gpu_train_autoencoder.json','configs/multi_gpu_train_diffusion.json']" --lr 8e-5 +``` + +#### Execute inference +The following code generates a synthetic image from a random sampled noise. +``` +python -m monai.bundle run --config_file configs/inference.json +``` + +#### Export checkpoint to TorchScript file + +The Autoencoder can be exported into a TorchScript file. + +``` +python -m monai.bundle ckpt_export autoencoder_def --filepath models/model_autoencoder.ts --ckpt_file models/model_autoencoder.pt --meta_file configs/metadata.json --config_file configs/inference.json +``` + +# References +[1] Rombach, Robin, et al. "High-resolution image synthesis with latent diffusion models." Proceedings of the IEEE/CVF Conference on Computer Vision and Pattern Recognition. 2022. https://openaccess.thecvf.com/content/CVPR2022/papers/Rombach_High-Resolution_Image_Synthesis_With_Latent_Diffusion_Models_CVPR_2022_paper.pdf + +# License +Copyright (c) MONAI Consortium + +Licensed under the Apache License, Version 2.0 (the "License"); +you may not use this file except in compliance with the License. +You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. diff --git a/models/brats_mri_generative_diffusion/docs/data_license.txt b/models/brats_mri_generative_diffusion/docs/data_license.txt new file mode 100644 index 00000000..d3d7e227 --- /dev/null +++ b/models/brats_mri_generative_diffusion/docs/data_license.txt @@ -0,0 +1,49 @@ +Third Party Licenses +----------------------------------------------------------------------- + +/*********************************************************************/ +i. Multimodal Brain Tumor Segmentation Challenge 2018 + https://www.med.upenn.edu/sbia/brats2018/data.html +/*********************************************************************/ + +Data Usage Agreement / Citations + +You are free to use and/or refer to the BraTS datasets in your own +research, provided that you always cite the following two manuscripts: + +[1] Menze BH, Jakab A, Bauer S, Kalpathy-Cramer J, Farahani K, Kirby +[J, Burren Y, Porz N, Slotboom J, Wiest R, Lanczi L, Gerstner E, Weber +[MA, Arbel T, Avants BB, Ayache N, Buendia P, Collins DL, Cordier N, +[Corso JJ, Criminisi A, Das T, Delingette H, Demiralp Γ, Durst CR, +[Dojat M, Doyle S, Festa J, Forbes F, Geremia E, Glocker B, Golland P, +[Guo X, Hamamci A, Iftekharuddin KM, Jena R, John NM, Konukoglu E, +[Lashkari D, Mariz JA, Meier R, Pereira S, Precup D, Price SJ, Raviv +[TR, Reza SM, Ryan M, Sarikaya D, Schwartz L, Shin HC, Shotton J, +[Silva CA, Sousa N, Subbanna NK, Szekely G, Taylor TJ, Thomas OM, +[Tustison NJ, Unal G, Vasseur F, Wintermark M, Ye DH, Zhao L, Zhao B, +[Zikic D, Prastawa M, Reyes M, Van Leemput K. "The Multimodal Brain +[Tumor Image Segmentation Benchmark (BRATS)", IEEE Transactions on +[Medical Imaging 34(10), 1993-2024 (2015) DOI: +[10.1109/TMI.2014.2377694 + +[2] Bakas S, Akbari H, Sotiras A, Bilello M, Rozycki M, Kirby JS, +[Freymann JB, Farahani K, Davatzikos C. "Advancing The Cancer Genome +[Atlas glioma MRI collections with expert segmentation labels and +[radiomic features", Nature Scientific Data, 4:170117 (2017) DOI: +[10.1038/sdata.2017.117 + +In addition, if there are no restrictions imposed from the +journal/conference you submit your paper about citing "Data +Citations", please be specific and also cite the following: + +[3] Bakas S, Akbari H, Sotiras A, Bilello M, Rozycki M, Kirby J, +[Freymann J, Farahani K, Davatzikos C. "Segmentation Labels and +[Radiomic Features for the Pre-operative Scans of the TCGA-GBM +[collection", The Cancer Imaging Archive, 2017. DOI: +[10.7937/K9/TCIA.2017.KLXWJJ1Q + +[4] Bakas S, Akbari H, Sotiras A, Bilello M, Rozycki M, Kirby J, +[Freymann J, Farahani K, Davatzikos C. "Segmentation Labels and +[Radiomic Features for the Pre-operative Scans of the TCGA-LGG +[collection", The Cancer Imaging Archive, 2017. DOI: +[10.7937/K9/TCIA.2017.GJQ7R0EF diff --git a/models/brats_mri_generative_diffusion/large_files.yml b/models/brats_mri_generative_diffusion/large_files.yml new file mode 100644 index 00000000..24af4605 --- /dev/null +++ b/models/brats_mri_generative_diffusion/large_files.yml @@ -0,0 +1,13 @@ +large_files: + - path: "models/model_autoencoder.pt" + url: "https://drive.google.com/uc?id=1arp3w8glsQw2h7mQBbk71krqmaG_std6" + hash_val: "314c9091a4b101461a5cc305995a4565" + hash_type: "md5" + - path: "models/model.pt" + url: "https://drive.google.com/uc?id=1m2pcbj8NMoxEIAOmD9dgYBN4gNcrMx6e" + hash_val: "2463887ef1b704494014e2aecb72802b" + hash_type: "md5" + - path: "models/model_autoencoder.ts" + url: "https://drive.google.com/uc?id=17WybvJHfxL-jc5toFFHxCfgVmFun-5ul" + hash_val: "8ad34a1080a71cf0a21964621a4f3e2e" + hash_type: "md5" diff --git a/models/brats_mri_generative_diffusion/scripts/__init__.py b/models/brats_mri_generative_diffusion/scripts/__init__.py new file mode 100644 index 00000000..2041a809 --- /dev/null +++ b/models/brats_mri_generative_diffusion/scripts/__init__.py @@ -0,0 +1,12 @@ +# Copyright (c) MONAI Consortium +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# http://www.apache.org/licenses/LICENSE-2.0 +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +from . import ldm_sampler, ldm_trainer, losses, utils diff --git a/models/brats_mri_generative_diffusion/scripts/ldm_sampler.py b/models/brats_mri_generative_diffusion/scripts/ldm_sampler.py new file mode 100644 index 00000000..392d3333 --- /dev/null +++ b/models/brats_mri_generative_diffusion/scripts/ldm_sampler.py @@ -0,0 +1,60 @@ +# Copyright (c) MONAI Consortium +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# http://www.apache.org/licenses/LICENSE-2.0 +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +from __future__ import annotations + +import torch +import torch.nn as nn +from monai.utils import optional_import +from torch.cuda.amp import autocast + +tqdm, has_tqdm = optional_import("tqdm", name="tqdm") + + +class LDMSampler: + def __init__(self) -> None: + super().__init__() + + @torch.no_grad() + def sampling_fn( + self, + input_noise: torch.Tensor, + autoencoder_model: nn.Module, + diffusion_model: nn.Module, + scheduler: nn.Module, + conditioning: torch.Tensor | None = None, + ) -> torch.Tensor: + if has_tqdm: + progress_bar = tqdm(scheduler.timesteps) + else: + progress_bar = iter(scheduler.timesteps) + + image = input_noise + if conditioning is not None: + cond_concat = conditioning.squeeze(1).unsqueeze(-1).unsqueeze(-1).unsqueeze(-1) + cond_concat = cond_concat.expand(list(cond_concat.shape[0:2]) + list(input_noise.shape[2:])) + + for t in progress_bar: + with torch.no_grad(): + if conditioning is not None: + input_t = torch.cat((image, cond_concat), dim=1) + else: + input_t = image + model_output = diffusion_model( + input_t, timesteps=torch.Tensor((t,)).to(input_noise.device).long(), context=conditioning + ) + image, _ = scheduler.step(model_output, t, image) + + with torch.no_grad(): + with autocast(): + sample = autoencoder_model.decode_stage_2_outputs(image) + + return sample diff --git a/models/brats_mri_generative_diffusion/scripts/ldm_trainer.py b/models/brats_mri_generative_diffusion/scripts/ldm_trainer.py new file mode 100644 index 00000000..c1a21bfa --- /dev/null +++ b/models/brats_mri_generative_diffusion/scripts/ldm_trainer.py @@ -0,0 +1,380 @@ +# Copyright (c) MONAI Consortium +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# http://www.apache.org/licenses/LICENSE-2.0 +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +from __future__ import annotations + +from typing import TYPE_CHECKING, Any, Callable, Iterable, Sequence + +import torch +from monai.config import IgniteInfo +from monai.engines.utils import IterationEvents, default_metric_cmp_fn, default_prepare_batch +from monai.inferers import Inferer, SimpleInferer +from monai.transforms import Transform +from monai.utils import min_version, optional_import +from monai.utils.enums import CommonKeys, GanKeys +from torch.optim.optimizer import Optimizer +from torch.utils.data import DataLoader + +if TYPE_CHECKING: + from ignite.engine import Engine, EventEnum + from ignite.metrics import Metric +else: + Engine, _ = optional_import("ignite.engine", IgniteInfo.OPT_IMPORT_VERSION, min_version, "Engine") + Metric, _ = optional_import("ignite.metrics", IgniteInfo.OPT_IMPORT_VERSION, min_version, "Metric") + EventEnum, _ = optional_import("ignite.engine", IgniteInfo.OPT_IMPORT_VERSION, min_version, "EventEnum") +from monai.engines.trainer import SupervisedTrainer, Trainer + + +class VaeGanTrainer(Trainer): + """ + Generative adversarial network training based on Goodfellow et al. 2014 https://arxiv.org/abs/1406.266, + inherits from ``Trainer`` and ``Workflow``. + Training Loop: for each batch of data size `m` + 1. Generate `m` fakes from random latent codes. + 2. Update discriminator with these fakes and current batch reals, repeated d_train_steps times. + 3. If g_update_latents, generate `m` fakes from new random latent codes. + 4. Update generator with these fakes using discriminator feedback. + Args: + device: an object representing the device on which to run. + max_epochs: the total epoch number for engine to run. + train_data_loader: Core ignite engines uses `DataLoader` for training loop batchdata. + g_network: generator (G) network architecture. + g_optimizer: G optimizer function. + g_loss_function: G loss function for optimizer. + d_network: discriminator (D) network architecture. + d_optimizer: D optimizer function. + d_loss_function: D loss function for optimizer. + epoch_length: number of iterations for one epoch, default to `len(train_data_loader)`. + g_inferer: inference method to execute G model forward. Defaults to ``SimpleInferer()``. + d_inferer: inference method to execute D model forward. Defaults to ``SimpleInferer()``. + d_train_steps: number of times to update D with real data minibatch. Defaults to ``1``. + latent_shape: size of G input latent code. Defaults to ``64``. + non_blocking: if True and this copy is between CPU and GPU, the copy may occur asynchronously + with respect to the host. For other cases, this argument has no effect. + d_prepare_batch: callback function to prepare batchdata for D inferer. + Defaults to return ``GanKeys.REALS`` in batchdata dict. for more details please refer to: + https://pytorch.org/ignite/generated/ignite.engine.create_supervised_trainer.html. + g_prepare_batch: callback function to create batch of latent input for G inferer. + Defaults to return random latents. for more details please refer to: + https://pytorch.org/ignite/generated/ignite.engine.create_supervised_trainer.html. + g_update_latents: Calculate G loss with new latent codes. Defaults to ``True``. + iteration_update: the callable function for every iteration, expect to accept `engine` + and `engine.state.batch` as inputs, return data will be stored in `engine.state.output`. + if not provided, use `self._iteration()` instead. for more details please refer to: + https://pytorch.org/ignite/generated/ignite.engine.engine.Engine.html. + postprocessing: execute additional transformation for the model output data. + Typically, several Tensor based transforms composed by `Compose`. + key_train_metric: compute metric when every iteration completed, and save average value to + engine.state.metrics when epoch completed. key_train_metric is the main metric to compare and save the + checkpoint into files. + additional_metrics: more Ignite metrics that also attach to Ignite Engine. + metric_cmp_fn: function to compare current key metric with previous best key metric value, + it must accept 2 args (current_metric, previous_best) and return a bool result: if `True`, will update + `best_metric` and `best_metric_epoch` with current metric and epoch, default to `greater than`. + train_handlers: every handler is a set of Ignite Event-Handlers, must have `attach` function, like: + CheckpointHandler, StatsHandler, etc. + decollate: whether to decollate the batch-first data to a list of data after model computation, + recommend `decollate=True` when `postprocessing` uses components from `monai.transforms`. + default to `True`. + optim_set_to_none: when calling `optimizer.zero_grad()`, instead of setting to zero, set the grads to None. + more details: https://pytorch.org/docs/stable/generated/torch.optim.Optimizer.zero_grad.html. + to_kwargs: dict of other args for `prepare_batch` API when converting the input data, except for + `device`, `non_blocking`. + amp_kwargs: dict of the args for `torch.cuda.amp.autocast()` API, for more details: + https://pytorch.org/docs/stable/amp.html#torch.cuda.amp.autocast. + """ + + def __init__( + self, + device: str | torch.device, + max_epochs: int, + train_data_loader: DataLoader, + g_network: torch.nn.Module, + g_optimizer: Optimizer, + g_loss_function: Callable, + d_network: torch.nn.Module, + d_optimizer: Optimizer, + d_loss_function: Callable, + epoch_length: int | None = None, + g_inferer: Inferer | None = None, + d_inferer: Inferer | None = None, + d_train_steps: int = 1, + latent_shape: int = 64, + non_blocking: bool = False, + d_prepare_batch: Callable = default_prepare_batch, + g_prepare_batch: Callable = default_prepare_batch, + g_update_latents: bool = True, + iteration_update: Callable[[Engine, Any], Any] | None = None, + postprocessing: Transform | None = None, + key_train_metric: dict[str, Metric] | None = None, + additional_metrics: dict[str, Metric] | None = None, + metric_cmp_fn: Callable = default_metric_cmp_fn, + train_handlers: Sequence | None = None, + decollate: bool = True, + optim_set_to_none: bool = False, + to_kwargs: dict | None = None, + amp_kwargs: dict | None = None, + ): + if not isinstance(train_data_loader, DataLoader): + raise ValueError("train_data_loader must be PyTorch DataLoader.") + + # set up Ignite engine and environments + super().__init__( + device=device, + max_epochs=max_epochs, + data_loader=train_data_loader, + epoch_length=epoch_length, + non_blocking=non_blocking, + prepare_batch=d_prepare_batch, + iteration_update=iteration_update, + key_metric=key_train_metric, + additional_metrics=additional_metrics, + metric_cmp_fn=metric_cmp_fn, + handlers=train_handlers, + postprocessing=postprocessing, + decollate=decollate, + to_kwargs=to_kwargs, + amp_kwargs=amp_kwargs, + ) + self.g_network = g_network + self.g_optimizer = g_optimizer + self.g_loss_function = g_loss_function + self.g_inferer = SimpleInferer() if g_inferer is None else g_inferer + self.d_network = d_network + self.d_optimizer = d_optimizer + self.d_loss_function = d_loss_function + self.d_inferer = SimpleInferer() if d_inferer is None else d_inferer + self.d_train_steps = d_train_steps + self.latent_shape = latent_shape + self.g_prepare_batch = g_prepare_batch + self.g_update_latents = g_update_latents + self.optim_set_to_none = optim_set_to_none + + def _iteration( + self, engine: VaeGanTrainer, batchdata: dict | Sequence + ) -> dict[str, torch.Tensor | int | float | bool]: + """ + Callback function for Adversarial Training processing logic of 1 iteration in Ignite Engine. + Args: + engine: `VaeGanTrainer` to execute operation for an iteration. + batchdata: input data for this iteration, usually can be dictionary or tuple of Tensor data. + Raises: + ValueError: must provide batch data for current iteration. + """ + if batchdata is None: + raise ValueError("must provide batch data for current iteration.") + + d_input = engine.prepare_batch(batchdata, engine.state.device, engine.non_blocking, **engine.to_kwargs)[0] + g_input = d_input + g_output, z_mu, z_sigma = engine.g_inferer(g_input, engine.g_network) + + # Train Discriminator + d_total_loss = torch.zeros(1) + for _ in range(engine.d_train_steps): + engine.d_optimizer.zero_grad(set_to_none=engine.optim_set_to_none) + dloss = engine.d_loss_function(g_output, d_input) + dloss.backward() + engine.d_optimizer.step() + d_total_loss += dloss.item() + + # Train Generator + engine.g_optimizer.zero_grad(set_to_none=engine.optim_set_to_none) + g_loss = engine.g_loss_function(g_output, g_input, z_mu, z_sigma) + g_loss.backward() + engine.g_optimizer.step() + + return { + GanKeys.REALS: d_input, + GanKeys.FAKES: g_output, + GanKeys.LATENTS: g_input, + GanKeys.GLOSS: g_loss.item(), + GanKeys.DLOSS: d_total_loss.item(), + } + + +class LDMTrainer(SupervisedTrainer): + """ + Standard supervised training method with image and label, inherits from ``Trainer`` and ``Workflow``. + Args: + device: an object representing the device on which to run. + max_epochs: the total epoch number for trainer to run. + train_data_loader: Ignite engine use data_loader to run, must be Iterable or torch.DataLoader. + network: network to train in the trainer, should be regular PyTorch `torch.nn.Module`. + optimizer: the optimizer associated to the network, should be regular PyTorch optimizer from `torch.optim` + or its subclass. + loss_function: the loss function associated to the optimizer, should be regular PyTorch loss, + which inherit from `torch.nn.modules.loss`. + epoch_length: number of iterations for one epoch, default to `len(train_data_loader)`. + non_blocking: if True and this copy is between CPU and GPU, the copy may occur asynchronously + with respect to the host. For other cases, this argument has no effect. + prepare_batch: function to parse expected data (usually `image`, `label` and other network args) + from `engine.state.batch` for every iteration, for more details please refer to: + https://pytorch.org/ignite/generated/ignite.engine.create_supervised_trainer.html. + iteration_update: the callable function for every iteration, expect to accept `engine` + and `engine.state.batch` as inputs, return data will be stored in `engine.state.output`. + if not provided, use `self._iteration()` instead. for more details please refer to: + https://pytorch.org/ignite/generated/ignite.engine.engine.Engine.html. + inferer: inference method that execute model forward on input data, like: SlidingWindow, etc. + postprocessing: execute additional transformation for the model output data. + Typically, several Tensor based transforms composed by `Compose`. + key_train_metric: compute metric when every iteration completed, and save average value to + engine.state.metrics when epoch completed. key_train_metric is the main metric to compare and save the + checkpoint into files. + additional_metrics: more Ignite metrics that also attach to Ignite Engine. + metric_cmp_fn: function to compare current key metric with previous best key metric value, + it must accept 2 args (current_metric, previous_best) and return a bool result: if `True`, will update + `best_metric` and `best_metric_epoch` with current metric and epoch, default to `greater than`. + train_handlers: every handler is a set of Ignite Event-Handlers, must have `attach` function, like: + CheckpointHandler, StatsHandler, etc. + amp: whether to enable auto-mixed-precision training, default is False. + event_names: additional custom ignite events that will register to the engine. + new events can be a list of str or `ignite.engine.events.EventEnum`. + event_to_attr: a dictionary to map an event to a state attribute, then add to `engine.state`. + for more details, check: https://pytorch.org/ignite/generated/ignite.engine.engine.Engine.html + #ignite.engine.engine.Engine.register_events. + decollate: whether to decollate the batch-first data to a list of data after model computation, + recommend `decollate=True` when `postprocessing` uses components from `monai.transforms`. + default to `True`. + optim_set_to_none: when calling `optimizer.zero_grad()`, instead of setting to zero, set the grads to None. + more details: https://pytorch.org/docs/stable/generated/torch.optim.Optimizer.zero_grad.html. + to_kwargs: dict of other args for `prepare_batch` API when converting the input data, except for + `device`, `non_blocking`. + amp_kwargs: dict of the args for `torch.cuda.amp.autocast()` API, for more details: + https://pytorch.org/docs/stable/amp.html#torch.cuda.amp.autocast. + """ + + def __init__( + self, + device: str | torch.device, + max_epochs: int, + train_data_loader: Iterable | DataLoader, + network: torch.nn.Module, + autoencoder_model: torch.nn.Module, + optimizer: Optimizer, + loss_function: Callable, + latent_shape: Sequence, + inferer: Inferer, + epoch_length: int | None = None, + non_blocking: bool = False, + prepare_batch: Callable = default_prepare_batch, + iteration_update: Callable[[Engine, Any], Any] | None = None, + postprocessing: Transform | None = None, + key_train_metric: dict[str, Metric] | None = None, + additional_metrics: dict[str, Metric] | None = None, + metric_cmp_fn: Callable = default_metric_cmp_fn, + train_handlers: Sequence | None = None, + amp: bool = False, + event_names: list[str | EventEnum | type[EventEnum]] | None = None, + event_to_attr: dict | None = None, + decollate: bool = True, + optim_set_to_none: bool = False, + to_kwargs: dict | None = None, + amp_kwargs: dict | None = None, + ) -> None: + super().__init__( + device=device, + max_epochs=max_epochs, + train_data_loader=train_data_loader, + network=network, + optimizer=optimizer, + loss_function=loss_function, + inferer=inferer, + optim_set_to_none=optim_set_to_none, + epoch_length=epoch_length, + non_blocking=non_blocking, + prepare_batch=prepare_batch, + iteration_update=iteration_update, + postprocessing=postprocessing, + key_train_metric=key_train_metric, + additional_metrics=additional_metrics, + metric_cmp_fn=metric_cmp_fn, + train_handlers=train_handlers, + amp=amp, + event_names=event_names, + event_to_attr=event_to_attr, + decollate=decollate, + to_kwargs=to_kwargs, + amp_kwargs=amp_kwargs, + ) + + self.latent_shape = latent_shape + self.autoencoder_model = autoencoder_model + + def _iteration(self, engine: LDMTrainer, batchdata: dict[str, torch.Tensor]) -> dict: + """ + Callback function for the Supervised Training processing logic of 1 iteration in Ignite Engine. + Return below items in a dictionary: + - IMAGE: image Tensor data for model input, already moved to device. + - LABEL: label Tensor data corresponding to the image, already moved to device. + - PRED: prediction result of model. + - LOSS: loss value computed by loss function. + Args: + engine: `SupervisedTrainer` to execute operation for an iteration. + batchdata: input data for this iteration, usually can be dictionary or tuple of Tensor data. + Raises: + ValueError: When ``batchdata`` is None. + """ + if batchdata is None: + raise ValueError("Must provide batch data for current iteration.") + batch = engine.prepare_batch(batchdata, engine.state.device, engine.non_blocking, **engine.to_kwargs) + if len(batch) == 2: + images, labels = batch + args: tuple = () + kwargs: dict = {} + else: + images, labels, args, kwargs = batch + # put iteration outputs into engine.state + engine.state.output = {CommonKeys.IMAGE: images} + + # generate noise + noise_shape = [images.shape[0]] + list(self.latent_shape) + noise = torch.randn(noise_shape, dtype=images.dtype).to(images.device) + engine.state.output = {"noise": noise} + + # Create timesteps + timesteps = torch.randint( + 0, engine.inferer.scheduler.num_train_timesteps, (images.shape[0],), device=images.device + ).long() + + def _compute_pred_loss(): + # predicted noise + engine.state.output[CommonKeys.PRED] = engine.inferer( + inputs=images, + autoencoder_model=self.autoencoder_model, + diffusion_model=engine.network, + noise=noise, + timesteps=timesteps, + ) + engine.fire_event(IterationEvents.FORWARD_COMPLETED) + # compute loss + engine.state.output[CommonKeys.LOSS] = engine.loss_function( + engine.state.output[CommonKeys.PRED], noise + ).mean() + engine.fire_event(IterationEvents.LOSS_COMPLETED) + + engine.network.train() + engine.optimizer.zero_grad(set_to_none=engine.optim_set_to_none) + + if engine.amp and engine.scaler is not None: + with torch.cuda.amp.autocast(**engine.amp_kwargs): + _compute_pred_loss() + engine.scaler.scale(engine.state.output[CommonKeys.LOSS]).backward() + engine.fire_event(IterationEvents.BACKWARD_COMPLETED) + engine.scaler.step(engine.optimizer) + engine.scaler.update() + else: + _compute_pred_loss() + engine.state.output[CommonKeys.LOSS].backward() + engine.fire_event(IterationEvents.BACKWARD_COMPLETED) + engine.optimizer.step() + engine.fire_event(IterationEvents.MODEL_COMPLETED) + + return engine.state.output diff --git a/models/brats_mri_generative_diffusion/scripts/losses.py b/models/brats_mri_generative_diffusion/scripts/losses.py new file mode 100644 index 00000000..43536067 --- /dev/null +++ b/models/brats_mri_generative_diffusion/scripts/losses.py @@ -0,0 +1,50 @@ +# Copyright (c) MONAI Consortium +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# http://www.apache.org/licenses/LICENSE-2.0 +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and + +import torch +from generative.losses import PatchAdversarialLoss + +intensity_loss = torch.nn.L1Loss() +adv_loss = PatchAdversarialLoss(criterion="least_squares") + +adv_weight = 0.1 +perceptual_weight = 0.1 +# kl_weight: important hyper-parameter. +# If too large, decoder cannot recon good results from latent space. +# If too small, latent space will not be regularized enough for the diffusion model +kl_weight = 1e-7 + + +def compute_kl_loss(z_mu, z_sigma): + kl_loss = 0.5 * torch.sum(z_mu.pow(2) + z_sigma.pow(2) - torch.log(z_sigma.pow(2)) - 1, dim=[1, 2, 3, 4]) + return torch.sum(kl_loss) / kl_loss.shape[0] + + +def generator_loss(gen_images, real_images, z_mu, z_sigma, disc_net, loss_perceptual): + recons_loss = intensity_loss(gen_images, real_images) + kl_loss = compute_kl_loss(z_mu, z_sigma) + p_loss = loss_perceptual(gen_images.float(), real_images.float()) + loss_g = recons_loss + kl_weight * kl_loss + perceptual_weight * p_loss + + logits_fake = disc_net(gen_images)[-1] + generator_loss = adv_loss(logits_fake, target_is_real=True, for_discriminator=False) + loss_g = loss_g + adv_weight * generator_loss + + return loss_g + + +def discriminator_loss(gen_images, real_images, disc_net): + logits_fake = disc_net(gen_images.contiguous().detach())[-1] + loss_d_fake = adv_loss(logits_fake, target_is_real=False, for_discriminator=True) + logits_real = disc_net(real_images.contiguous().detach())[-1] + loss_d_real = adv_loss(logits_real, target_is_real=True, for_discriminator=True) + discriminator_loss = (loss_d_fake + loss_d_real) * 0.5 + loss_d = adv_weight * discriminator_loss + return loss_d diff --git a/models/brats_mri_generative_diffusion/scripts/utils.py b/models/brats_mri_generative_diffusion/scripts/utils.py new file mode 100644 index 00000000..57a6744e --- /dev/null +++ b/models/brats_mri_generative_diffusion/scripts/utils.py @@ -0,0 +1,20 @@ +# Copyright (c) MONAI Consortium +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# http://www.apache.org/licenses/LICENSE-2.0 +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and + +import monai +import torch + + +def compute_scale_factor(autoencoder, train_loader, device): + with torch.no_grad(): + check_data = monai.utils.first(train_loader) + z = autoencoder.encode_stage_2_inputs(check_data["image"].to(device)) + scale_factor = 1 / torch.std(z) + return scale_factor.item() diff --git a/pyproject.toml b/pyproject.toml index ac0df5bf..223ea491 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -43,7 +43,7 @@ check_parameter_types = true # Check variable values against their annotations. check_variable_types = true # Comma or space separated list of error names to ignore. -disable = ["pyi-error"] +disable = ["pyi-error", "import-error"] # Report errors. report_errors = true # Experimental: Infer precise return types even for invalid function calls.