Source code for anomalib.utils.callbacks.visualizer_callback
"""Visualizer Callback."""# Copyright (C) 2020 Intel Corporation## 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.frompathlibimportPathfromtypingimportAny,Optional,castimportnumpyasnpimportpytorch_lightningasplfrompytorch_lightningimportCallbackfrompytorch_lightning.utilities.cliimportCALLBACK_REGISTRYfrompytorch_lightning.utilities.typesimportSTEP_OUTPUTfromanomalib.models.componentsimportAnomalyModulefromanomalib.post_processingimportVisualizerfromanomalib.utils.loggersimportAnomalibWandbLoggerfromanomalib.utils.loggers.baseimportImageLoggerBase@CALLBACK_REGISTRY
[docs]classVisualizerCallback(Callback):"""Callback that visualizes the inference results of a model. The callback generates a figure showing the original image, the ground truth segmentation mask, the predicted error heat map, and the predicted segmentation mask. To save the images to the filesystem, add the 'local' keyword to the `project.log_images_to` parameter in the config.yaml file. """def__init__(self,task:str,mode:str,image_save_path:str,inputs_are_normalized:bool=True,show_images:bool=False,log_images:bool=True,save_images:bool=True,):"""Visualizer callback."""ifmodenotin["full","simple"]:raiseValueError(f"Unknown visualization mode: {mode}. Please choose one of ['full', 'simple']")self.mode=modeiftasknotin["classification","segmentation"]:raiseValueError(f"Unknown task type: {mode}. Please choose one of ['classification', 'segmentation']")self.task=taskself.inputs_are_normalized=inputs_are_normalizedself.show_images=show_imagesself.log_images=log_imagesself.save_images=save_imagesself.image_save_path=Path(image_save_path)self.visualizer=Visualizer(mode,task)
[docs]def_add_to_logger(self,image:np.ndarray,module:AnomalyModule,trainer:pl.Trainer,filename:Path,):"""Log image from a visualizer to each of the available loggers in the project. Args: image (np.ndarray): Image that should be added to the loggers. module (AnomalyModule): Anomaly module. trainer (Trainer): Pytorch Lightning trainer which holds reference to `logger` filename (Path): Path of the input image. This name is used as name for the generated image. """# Store names of logger and the logger in a dictavailable_loggers={type(logger).__name__.lower().rstrip("logger").lstrip("anomalib"):loggerforloggerintrainer.loggers}# save image to respective loggerifself.log_images:forlog_toinavailable_loggers:# check if logger object is same as the requested objectifisinstance(available_loggers[log_to],ImageLoggerBase):logger:ImageLoggerBase=cast(ImageLoggerBase,available_loggers[log_to])# placate mypylogger.add_image(image=image,name=filename.parent.name+"_"+filename.name,global_step=module.global_step,
)
[docs]defon_predict_batch_end(self,_trainer:pl.Trainer,_pl_module:AnomalyModule,outputs:Optional[STEP_OUTPUT],_batch:Any,_batch_idx:int,_dataloader_idx:int,)->None:"""Show images at the end of every batch. Args: _trainer (Trainer): Pytorch lightning trainer object (unused). _pl_module (LightningModule): Lightning modules derived from BaseAnomalyLightning object as currently only they support logging images. outputs (Dict[str, Any]): Outputs of the current test step. _batch (Any): Input batch of the current test step (unused). _batch_idx (int): Index of the current test batch (unused). _dataloader_idx (int): Index of the dataloader that yielded the current batch (unused). """assertoutputsisnotNonefori,imageinenumerate(self.visualizer.visualize_batch(outputs)):filename=Path(outputs["image_path"][i])ifself.save_images:file_path=self.image_save_path/filename.parent.name/filename.nameself.visualizer.save(file_path,image)ifself.show_images:self.visualizer.show(str(filename),image)
[docs]defon_test_batch_end(self,trainer:pl.Trainer,pl_module:AnomalyModule,outputs:Optional[STEP_OUTPUT],_batch:Any,_batch_idx:int,_dataloader_idx:int,)->None:"""Log images at the end of every batch. Args: trainer (Trainer): Pytorch lightning trainer object (unused). pl_module (LightningModule): Lightning modules derived from BaseAnomalyLightning object as currently only they support logging images. outputs (Dict[str, Any]): Outputs of the current test step. _batch (Any): Input batch of the current test step (unused). _batch_idx (int): Index of the current test batch (unused). _dataloader_idx (int): Index of the dataloader that yielded the current batch (unused). """assertoutputsisnotNonefori,imageinenumerate(self.visualizer.visualize_batch(outputs)):filename=Path(outputs["image_path"][i])ifself.save_images:file_path=self.image_save_path/filename.parent.name/filename.nameself.visualizer.save(file_path,image)ifself.log_images:self._add_to_logger(image,pl_module,trainer,filename)ifself.show_images:self.visualizer.show(str(filename),image)
[docs]defon_test_end(self,_trainer:pl.Trainer,pl_module:AnomalyModule)->None:"""Sync logs. Currently only ``AnomalibWandbLogger`` is called from this method. This is because logging as a single batch ensures that all images appear as part of the same step. Args: _trainer (pl.Trainer): Pytorch Lightning trainer (unused) pl_module (AnomalyModule): Anomaly module """ifpl_module.loggerisnotNoneandisinstance(pl_module.logger,AnomalibWandbLogger):pl_module.logger.save()