hippo-nn


Namehippo-nn JSON
Version 0.1.0 PyPI version JSON
download
home_pageNone
SummaryHIPPO explainability toolkit for computational pathology.
upload_time2024-09-06 11:54:06
maintainerNone
docs_urlNone
authorNone
requires_python>=3.8
licenseBSD 3-Clause License Copyright (c) 2024, Jakub Kaczmarzyk Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met: 1. Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. 2. Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution. 3. Neither the name of the copyright holder nor the names of its contributors may be used to endorse or promote products derived from this software without specific prior written permission. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
keywords abmil computational pathology counterfactuals deep learning explainability multiple instance learning pathology
VCS
bugtrack_url
requirements No requirements were recorded.
Travis-CI No Travis.
coveralls test coverage No coveralls.
            # HIPPO

HIPPO is an explainability toolkit for weakly-supervised learning in computational pathology.

Please see our preprint on arXiv https://arxiv.org/abs/2409.03080.

> [!NOTE]
> This codebase is a work in progress. Please check back periodically for updates.

Abstract
--------
<img src="docs/logo.png" width="200px" align="right" />

Deep learning models have shown promise in histopathology image analysis, but their opaque decision-making process poses challenges in high-risk medical scenarios. Here we introduce HIPPO, an explainable AI method that interrogates attention-based multiple instance learning (ABMIL) models in computational pathology by generating counterfactual examples through tissue patch modifications in whole slide images. Applying HIPPO to ABMIL models trained to detect breast cancer metastasis reveals that they may overlook small tumors and can be misled by non-tumor tissue, while attention maps—widely used for interpretation—often highlight regions that do not directly influence predictions. By interpreting ABMIL models trained on a prognostic prediction task, HIPPO identified tissue areas with stronger prognostic effects than high-attention regions, which sometimes showed counterintuitive influences on risk scores. These findings demonstrate HIPPO's capacity for comprehensive model evaluation, bias detection, and quantitative hypothesis testing. HIPPO greatly expands the capabilities of explainable AI tools to assess the trustworthy and reliable development, deployment, and regulation of weakly-supervised models in computational pathology.

If you find HIPPO useful, kindly [cite](#cite) it in your work.

# How to use HIPPO

HIPPO is meant for weakly-supervised, multiple instance learning models in computational pathology. Before you use HIPPO, you need patch embeddings, and and a trained attention-based multiple instance learning (ABMIL) model. Below, we briefly describe how to go from whole slide images (WSIs) to a trained ABMIL model.

We also made available models for metastasis detection, trained on CAMELYON16. Please see the following HuggingFace repositories for metastasis detection models trained using different encoders:
- UNI: https://huggingface.co/kaczmarj/metastasis-abmil-128um-uni
- REMEDIS: https://huggingface.co/kaczmarj/metastasis-abmil-128um-remedis
- Phikon: https://huggingface.co/kaczmarj/metastasis-abmil-128um-phikon
- CTransPath: https://huggingface.co/kaczmarj/metastasis-abmil-128um-ctranspath
- RetCCL: https://huggingface.co/kaczmarj/metastasis-abmil-128um-retccl

To simplify reproducibility, we also uploaded UNI embeddings for CAMELYON16 to https://huggingface.co/datasets/kaczmarj/camelyon16-uni. Embeddings using the other models may be uploaded in the future.

## Prepare your data for ABMIL

First separate your whole slide images into smaller, non-overlapping patches. The CLAM toolkit is one popular way to do this. After you have patch coordinates, you will have to encode those patches with a pre-trained model. There are countless options to choose from, but I would opt for a recent foundation model trained on a large and diverse set of histopathology images. Keep track of the patch coordinates and the patch features. This will be useful for downstream HIPPO experiments and visualizing attention maps.

## Train the ABMIL model

We provide a training script for classification models at https://huggingface.co/kaczmarj/metastasis-abmil-128um-uni/blob/main/train_classification.py. Alternatively, trained a model with [CLAM](https://github.com/mahmoodlab/CLAM) or another toolkit. HIPPO can work with any weakly-supervised model that accepts a bag of patches and returns a specimen-level output.

# Examples

## Minimal reproducubile example with synthetic data

The code below isn't intended to show any effect of an intervention. Rather, the purpose is to show how to use HIPPO to create an intervention in a specimen and evaluate the effects using a pretrained ABMIL model.

To work with real data and a pretrained model, see [the example below](#test-the-sufficiency-of-tumor-for-metastasis-detection).


```python
import hippo
import numpy as np
import torch

# Create the ABMIL model. Here, we use random initializations for the example.
# You should use a pretrained model in practice.
model = hippo.AttentionMILModel(in_features=1024, L=512, D=384, num_classes=2)
model.eval()

# We use random features. In practice, use actual features :)
features = torch.rand(1000, 1024)

# Define the intervention. Here, we want to remove five patches.
# We define the indices of the patches to keep.
patches_to_remove = np.array([500, 501, 502, 503, 504])
patches_to_keep = np.setdiff1d(np.arange(features.shape[0]), patches_to_remove)

# Get the model outputs for baseline and "treated" samples.
with torch.inference_mode():
    baseline = model(features).logits.softmax(1)
    treatment = model(features[patches_to_keep]).logits.softmax(1)
```

## Test the sufficiency of tumor for metastasis detection

In the example below, we load a UNI-based ABMIL model for metastasis detection, trained on CAMELYON16.
Then, we take the embedding from one tumor patch from specimen `test_001` and add it to a negative specimen `test_003`.
The addition of this single tumor patch is enough to cause a positive metastasis result.

```python
import hippo
import huggingface_hub
import numpy as np
import torch

# Create the ABMIL model. Here, we use random initializations for the example.
# You should use a pretrained model in practice.
model = hippo.AttentionMILModel(in_features=1024, L=512, D=384, num_classes=2)
model.eval()
# You may need to run huggingface_hub.login() to get this file.
state_dict_path = huggingface_hub.hf_hub_download(
    "kaczmarj/metastasis-abmil-128um-uni", filename="seed2/model_best.pt"
)
state_dict = torch.load(state_dict_path, map_location="cpu", weights_only=True)
model.load_state_dict(state_dict)

features_positive_path = huggingface_hub.hf_hub_download(
    "kaczmarj/camelyon16-uni", filename="embeddings/test_001.pt", repo_type="dataset"
)
features_positive = torch.load(features_positive_path, weights_only=True)
# This index contains the embedding for the tumor patch shown in Figure 2a of the HIPPO preprint.
tumor_patch = features_positive[7238].unsqueeze(0)  # 1x1024

features_negative_patch = huggingface_hub.hf_hub_download(
    "kaczmarj/camelyon16-uni", filename="embeddings/test_003.pt", repo_type="dataset"
)
features_negative = torch.load(features_negative_patch, weights_only=True)

# Get the model outputs for baseline and treated samples.
with torch.inference_mode():
    baseline = model(features_negative).logits.softmax(1)[0, 1].item()
    treatment = model(torch.cat([features_negative, tumor_patch])).logits.softmax(1)[0, 1].item()

print(f"Probability of tumor in baseline: {baseline:0.3f}")  # 0.002
print(f"Probability of tumor after adding one tumor patch: {treatment:0.3f}")  # 0.824
```

## Test the effect of high attention regions

In this example, we evaluate the effect of high attention regions on metastasis detection. We find the following:

1. Using the original specimen, the model strongly predicts presence of metastasis (probability 0.997).
2. If we remove the top 1% of attended patches, the probability remains high for metastasis (0.988). This is presumably because some tumor patches remain in the specimen after removing top 1% of attention.
3. If we remove 5% of attention, then the probability of metastasis falls to 0.001.

In this way, we can quantify the effect of high attention regions.

```python
import math
import hippo
import huggingface_hub
import torch

# Create the ABMIL model. Here, we use random initializations for the example.
# You should use a pretrained model in practice.
model = hippo.AttentionMILModel(in_features=1024, L=512, D=384, num_classes=2)
model.eval()
# You may need to run huggingface_hub.login() to get this file.
state_dict_path = huggingface_hub.hf_hub_download(
    "kaczmarj/metastasis-abmil-128um-uni", filename="seed2/model_best.pt"
)
state_dict = torch.load(state_dict_path, map_location="cpu", weights_only=True)
model.load_state_dict(state_dict)

# Load features for positive specimen.
features_path = huggingface_hub.hf_hub_download(
    "kaczmarj/camelyon16-uni", filename="embeddings/test_001.pt", repo_type="dataset"
)
features = torch.load(features_path, weights_only=True)

# Get the model outputs for baseline and treated samples.
with torch.inference_mode():
    logits, attn = model(features)
attn = attn.squeeze(1).numpy()  # flatten tensor
tumor_prob = logits.softmax(1)[0, 1].item()
print(f"Tumor probability at baseline: {tumor_prob:0.3f}")

inds = attn.argsort()[::-1].copy()  # indices high to low, and copy to please torch
num_patches = math.ceil(len(inds) * 0.01)
with torch.inference_mode():
    logits_01pct, _ = model(features[inds[num_patches:]])
tumor_prob_01pct = logits_01pct.softmax(1)[0, 1].item()
print(f"Tumor probability after removing top 1% of attention: {tumor_prob_01pct:0.3f}")

num_patches = math.ceil(len(inds) * 0.05)
with torch.inference_mode():
    logits_05pct, _ = model(features[inds[num_patches:]])
tumor_prob_05pct = logits_05pct.softmax(1)[0, 1].item()
print(f"Tumor probability after removing top 5% of attention: {tumor_prob_05pct:0.3f}")
```

The following is printed:

```
Tumor probability at baseline: 0.997
Tumor probability after removing top 1% of attention: 0.988
Tumor probability after removing top 5% of attention: 0.001
```

## HIPPO greedy search algorithms

HIPPO implements greedy search algorithms to identify important patches. Below, we search for the patches that have the highest effect on metastasis detection. Briefly, we identify the patches that, when removed, result in the lowest probabilities for metastasis detections.

```python
import math
import hippo
import huggingface_hub
import numpy as np
import torch

# Set our device.
device = torch.device("cpu")
# device = torch.device("cuda")  # Uncomment if you have a GPU.
# device = torch.device("mps")  # Uncomment if you have an ARM Apple computer.

# Load ABMIL model.
model = hippo.AttentionMILModel(in_features=1024, L=512, D=384, num_classes=2)
model.eval()
# You may need to run huggingface_hub.login() to get this file.
state_dict_path = huggingface_hub.hf_hub_download(
    "kaczmarj/metastasis-abmil-128um-uni", filename="seed2/model_best.pt"
)
state_dict = torch.load(state_dict_path, map_location="cpu", weights_only=True)
model.load_state_dict(state_dict)
model.to(device)

# Load features.
features_path = huggingface_hub.hf_hub_download(
    "kaczmarj/camelyon16-uni", filename="embeddings/test_064.pt", repo_type="dataset"
)
features = torch.load(features_path, weights_only=True).to(device)


# Define a function that takes in a bag of features and returns model probabilities.
# The output values are the values we want to optimize during our search.
# This is why we use a function -- models can have different outputs. By defining
# a function that returns the values we want to optimize on, we can streamline the code.
def model_probs_fn(features):
    with torch.inference_mode():
        logits, _ = model(features)
    # Shape of logits is 1xC, where C is number of classes.
    probs = logits.softmax(1).squeeze(0)  # C
    return probs


# Find the 1% highest effect patches. These are the patches that, when removed, drop the probability
# of metastasis the most. The `results` variable is a dictionary with.... results of the search!
# The model outputs in `results["model_outputs"]` correspond to the results after removing the patches
# in `results["ablated_patches"][:k]`.
num_rounds = math.ceil(len(features) * 0.01)
results = hippo.greedy_search(
    features=features,
    model_probs_fn=model_probs_fn,
    num_rounds=num_rounds,
    output_index_to_optimize=1,
    # We use minimize because we want to minimize the model outputs
    # when the patches are *removed*.
    optimizer=hippo.minimize,
)

# Now we can test the effect of removing the 1% highest effect patches.
patches_not_ablated = np.setdiff1d(np.arange(len(features)), results["ablated_patches"])
with torch.inference_mode():
    prob_baseline = model(features).logits.softmax(1)[0, 1].item()  # 1.000
    prob_without_high_effect = model(features[patches_not_ablated]).logits.softmax(1)[0, 1].item()  # 0.008

print(f"Probability of metastasis at baseline: {prob_baseline:0.3f}")
print(f"Probability of metastasis after removing 1% highest effect patches: {prob_without_high_effect:0.3f}")
```

We can also plot the model outputs as we remove high effect patches, and we hope to see a monotonically decreasing line.

```python
import matplotlib.pyplot as plt
import numpy as np

model_results = results["model_outputs"][:, results["optimized_class_index"]]
plt.plot(model_results)
plt.xlabel("Number of patches removed")
plt.ylabel("Probability of metastasis")
```


# Cite

```bibtex
@misc{kaczmarzyk2024explainableaicomputationalpathology,
      title={Explainable AI for computational pathology identifies model limitations and tissue biomarkers},
      author={Jakub R. Kaczmarzyk and Joel H. Saltz and Peter K. Koo},
      year={2024},
      eprint={2409.03080},
      archivePrefix={arXiv},
      primaryClass={q-bio.TO},
      url={https://arxiv.org/abs/2409.03080},
}
```

# License

HIPPO code is licensed under the terms of the 3-Clause BSD License, and documentation is published under the terms of the Creative Commons Attribution-NonCommercial-ShareAlike 4.0 International copyright license (CC BY-NC-SA 4.0).

            

Raw data

            {
    "_id": null,
    "home_page": null,
    "name": "hippo-nn",
    "maintainer": null,
    "docs_url": null,
    "requires_python": ">=3.8",
    "maintainer_email": "Jakub Kaczmarzyk <jakub.kaczmarzyk@gmail.com>",
    "keywords": "abmil, computational pathology, counterfactuals, deep learning, explainability, multiple instance learning, pathology",
    "author": null,
    "author_email": "Jakub Kaczmarzyk <jakub.kaczmarzyk@gmail.com>",
    "download_url": "https://files.pythonhosted.org/packages/25/67/ec16254fc68594e4079f2bf4f06b2461c143e669e3fe3878485fb8ad8d1c/hippo_nn-0.1.0.tar.gz",
    "platform": null,
    "description": "# HIPPO\n\nHIPPO is an explainability toolkit for weakly-supervised learning in computational pathology.\n\nPlease see our preprint on arXiv https://arxiv.org/abs/2409.03080.\n\n> [!NOTE]\n> This codebase is a work in progress. Please check back periodically for updates.\n\nAbstract\n--------\n<img src=\"docs/logo.png\" width=\"200px\" align=\"right\" />\n\nDeep learning models have shown promise in histopathology image analysis, but their opaque decision-making process poses challenges in high-risk medical scenarios. Here we introduce HIPPO, an explainable AI method that interrogates attention-based multiple instance learning (ABMIL) models in computational pathology by generating counterfactual examples through tissue patch modifications in whole slide images. Applying HIPPO to ABMIL models trained to detect breast cancer metastasis reveals that they may overlook small tumors and can be misled by non-tumor tissue, while attention maps\u2014widely used for interpretation\u2014often highlight regions that do not directly influence predictions. By interpreting ABMIL models trained on a prognostic prediction task, HIPPO identified tissue areas with stronger prognostic effects than high-attention regions, which sometimes showed counterintuitive influences on risk scores. These findings demonstrate HIPPO's capacity for comprehensive model evaluation, bias detection, and quantitative hypothesis testing. HIPPO greatly expands the capabilities of explainable AI tools to assess the trustworthy and reliable development, deployment, and regulation of weakly-supervised models in computational pathology.\n\nIf you find HIPPO useful, kindly [cite](#cite) it in your work.\n\n# How to use HIPPO\n\nHIPPO is meant for weakly-supervised, multiple instance learning models in computational pathology. Before you use HIPPO, you need patch embeddings, and and a trained attention-based multiple instance learning (ABMIL) model. Below, we briefly describe how to go from whole slide images (WSIs) to a trained ABMIL model.\n\nWe also made available models for metastasis detection, trained on CAMELYON16. Please see the following HuggingFace repositories for metastasis detection models trained using different encoders:\n- UNI: https://huggingface.co/kaczmarj/metastasis-abmil-128um-uni\n- REMEDIS: https://huggingface.co/kaczmarj/metastasis-abmil-128um-remedis\n- Phikon: https://huggingface.co/kaczmarj/metastasis-abmil-128um-phikon\n- CTransPath: https://huggingface.co/kaczmarj/metastasis-abmil-128um-ctranspath\n- RetCCL: https://huggingface.co/kaczmarj/metastasis-abmil-128um-retccl\n\nTo simplify reproducibility, we also uploaded UNI embeddings for CAMELYON16 to https://huggingface.co/datasets/kaczmarj/camelyon16-uni. Embeddings using the other models may be uploaded in the future.\n\n## Prepare your data for ABMIL\n\nFirst separate your whole slide images into smaller, non-overlapping patches. The CLAM toolkit is one popular way to do this. After you have patch coordinates, you will have to encode those patches with a pre-trained model. There are countless options to choose from, but I would opt for a recent foundation model trained on a large and diverse set of histopathology images. Keep track of the patch coordinates and the patch features. This will be useful for downstream HIPPO experiments and visualizing attention maps.\n\n## Train the ABMIL model\n\nWe provide a training script for classification models at https://huggingface.co/kaczmarj/metastasis-abmil-128um-uni/blob/main/train_classification.py. Alternatively, trained a model with [CLAM](https://github.com/mahmoodlab/CLAM) or another toolkit. HIPPO can work with any weakly-supervised model that accepts a bag of patches and returns a specimen-level output.\n\n# Examples\n\n## Minimal reproducubile example with synthetic data\n\nThe code below isn't intended to show any effect of an intervention. Rather, the purpose is to show how to use HIPPO to create an intervention in a specimen and evaluate the effects using a pretrained ABMIL model.\n\nTo work with real data and a pretrained model, see [the example below](#test-the-sufficiency-of-tumor-for-metastasis-detection).\n\n\n```python\nimport hippo\nimport numpy as np\nimport torch\n\n# Create the ABMIL model. Here, we use random initializations for the example.\n# You should use a pretrained model in practice.\nmodel = hippo.AttentionMILModel(in_features=1024, L=512, D=384, num_classes=2)\nmodel.eval()\n\n# We use random features. In practice, use actual features :)\nfeatures = torch.rand(1000, 1024)\n\n# Define the intervention. Here, we want to remove five patches.\n# We define the indices of the patches to keep.\npatches_to_remove = np.array([500, 501, 502, 503, 504])\npatches_to_keep = np.setdiff1d(np.arange(features.shape[0]), patches_to_remove)\n\n# Get the model outputs for baseline and \"treated\" samples.\nwith torch.inference_mode():\n    baseline = model(features).logits.softmax(1)\n    treatment = model(features[patches_to_keep]).logits.softmax(1)\n```\n\n## Test the sufficiency of tumor for metastasis detection\n\nIn the example below, we load a UNI-based ABMIL model for metastasis detection, trained on CAMELYON16.\nThen, we take the embedding from one tumor patch from specimen `test_001` and add it to a negative specimen `test_003`.\nThe addition of this single tumor patch is enough to cause a positive metastasis result.\n\n```python\nimport hippo\nimport huggingface_hub\nimport numpy as np\nimport torch\n\n# Create the ABMIL model. Here, we use random initializations for the example.\n# You should use a pretrained model in practice.\nmodel = hippo.AttentionMILModel(in_features=1024, L=512, D=384, num_classes=2)\nmodel.eval()\n# You may need to run huggingface_hub.login() to get this file.\nstate_dict_path = huggingface_hub.hf_hub_download(\n    \"kaczmarj/metastasis-abmil-128um-uni\", filename=\"seed2/model_best.pt\"\n)\nstate_dict = torch.load(state_dict_path, map_location=\"cpu\", weights_only=True)\nmodel.load_state_dict(state_dict)\n\nfeatures_positive_path = huggingface_hub.hf_hub_download(\n    \"kaczmarj/camelyon16-uni\", filename=\"embeddings/test_001.pt\", repo_type=\"dataset\"\n)\nfeatures_positive = torch.load(features_positive_path, weights_only=True)\n# This index contains the embedding for the tumor patch shown in Figure 2a of the HIPPO preprint.\ntumor_patch = features_positive[7238].unsqueeze(0)  # 1x1024\n\nfeatures_negative_patch = huggingface_hub.hf_hub_download(\n    \"kaczmarj/camelyon16-uni\", filename=\"embeddings/test_003.pt\", repo_type=\"dataset\"\n)\nfeatures_negative = torch.load(features_negative_patch, weights_only=True)\n\n# Get the model outputs for baseline and treated samples.\nwith torch.inference_mode():\n    baseline = model(features_negative).logits.softmax(1)[0, 1].item()\n    treatment = model(torch.cat([features_negative, tumor_patch])).logits.softmax(1)[0, 1].item()\n\nprint(f\"Probability of tumor in baseline: {baseline:0.3f}\")  # 0.002\nprint(f\"Probability of tumor after adding one tumor patch: {treatment:0.3f}\")  # 0.824\n```\n\n## Test the effect of high attention regions\n\nIn this example, we evaluate the effect of high attention regions on metastasis detection. We find the following:\n\n1. Using the original specimen, the model strongly predicts presence of metastasis (probability 0.997).\n2. If we remove the top 1% of attended patches, the probability remains high for metastasis (0.988). This is presumably because some tumor patches remain in the specimen after removing top 1% of attention.\n3. If we remove 5% of attention, then the probability of metastasis falls to 0.001.\n\nIn this way, we can quantify the effect of high attention regions.\n\n```python\nimport math\nimport hippo\nimport huggingface_hub\nimport torch\n\n# Create the ABMIL model. Here, we use random initializations for the example.\n# You should use a pretrained model in practice.\nmodel = hippo.AttentionMILModel(in_features=1024, L=512, D=384, num_classes=2)\nmodel.eval()\n# You may need to run huggingface_hub.login() to get this file.\nstate_dict_path = huggingface_hub.hf_hub_download(\n    \"kaczmarj/metastasis-abmil-128um-uni\", filename=\"seed2/model_best.pt\"\n)\nstate_dict = torch.load(state_dict_path, map_location=\"cpu\", weights_only=True)\nmodel.load_state_dict(state_dict)\n\n# Load features for positive specimen.\nfeatures_path = huggingface_hub.hf_hub_download(\n    \"kaczmarj/camelyon16-uni\", filename=\"embeddings/test_001.pt\", repo_type=\"dataset\"\n)\nfeatures = torch.load(features_path, weights_only=True)\n\n# Get the model outputs for baseline and treated samples.\nwith torch.inference_mode():\n    logits, attn = model(features)\nattn = attn.squeeze(1).numpy()  # flatten tensor\ntumor_prob = logits.softmax(1)[0, 1].item()\nprint(f\"Tumor probability at baseline: {tumor_prob:0.3f}\")\n\ninds = attn.argsort()[::-1].copy()  # indices high to low, and copy to please torch\nnum_patches = math.ceil(len(inds) * 0.01)\nwith torch.inference_mode():\n    logits_01pct, _ = model(features[inds[num_patches:]])\ntumor_prob_01pct = logits_01pct.softmax(1)[0, 1].item()\nprint(f\"Tumor probability after removing top 1% of attention: {tumor_prob_01pct:0.3f}\")\n\nnum_patches = math.ceil(len(inds) * 0.05)\nwith torch.inference_mode():\n    logits_05pct, _ = model(features[inds[num_patches:]])\ntumor_prob_05pct = logits_05pct.softmax(1)[0, 1].item()\nprint(f\"Tumor probability after removing top 5% of attention: {tumor_prob_05pct:0.3f}\")\n```\n\nThe following is printed:\n\n```\nTumor probability at baseline: 0.997\nTumor probability after removing top 1% of attention: 0.988\nTumor probability after removing top 5% of attention: 0.001\n```\n\n## HIPPO greedy search algorithms\n\nHIPPO implements greedy search algorithms to identify important patches. Below, we search for the patches that have the highest effect on metastasis detection. Briefly, we identify the patches that, when removed, result in the lowest probabilities for metastasis detections.\n\n```python\nimport math\nimport hippo\nimport huggingface_hub\nimport numpy as np\nimport torch\n\n# Set our device.\ndevice = torch.device(\"cpu\")\n# device = torch.device(\"cuda\")  # Uncomment if you have a GPU.\n# device = torch.device(\"mps\")  # Uncomment if you have an ARM Apple computer.\n\n# Load ABMIL model.\nmodel = hippo.AttentionMILModel(in_features=1024, L=512, D=384, num_classes=2)\nmodel.eval()\n# You may need to run huggingface_hub.login() to get this file.\nstate_dict_path = huggingface_hub.hf_hub_download(\n    \"kaczmarj/metastasis-abmil-128um-uni\", filename=\"seed2/model_best.pt\"\n)\nstate_dict = torch.load(state_dict_path, map_location=\"cpu\", weights_only=True)\nmodel.load_state_dict(state_dict)\nmodel.to(device)\n\n# Load features.\nfeatures_path = huggingface_hub.hf_hub_download(\n    \"kaczmarj/camelyon16-uni\", filename=\"embeddings/test_064.pt\", repo_type=\"dataset\"\n)\nfeatures = torch.load(features_path, weights_only=True).to(device)\n\n\n# Define a function that takes in a bag of features and returns model probabilities.\n# The output values are the values we want to optimize during our search.\n# This is why we use a function -- models can have different outputs. By defining\n# a function that returns the values we want to optimize on, we can streamline the code.\ndef model_probs_fn(features):\n    with torch.inference_mode():\n        logits, _ = model(features)\n    # Shape of logits is 1xC, where C is number of classes.\n    probs = logits.softmax(1).squeeze(0)  # C\n    return probs\n\n\n# Find the 1% highest effect patches. These are the patches that, when removed, drop the probability\n# of metastasis the most. The `results` variable is a dictionary with.... results of the search!\n# The model outputs in `results[\"model_outputs\"]` correspond to the results after removing the patches\n# in `results[\"ablated_patches\"][:k]`.\nnum_rounds = math.ceil(len(features) * 0.01)\nresults = hippo.greedy_search(\n    features=features,\n    model_probs_fn=model_probs_fn,\n    num_rounds=num_rounds,\n    output_index_to_optimize=1,\n    # We use minimize because we want to minimize the model outputs\n    # when the patches are *removed*.\n    optimizer=hippo.minimize,\n)\n\n# Now we can test the effect of removing the 1% highest effect patches.\npatches_not_ablated = np.setdiff1d(np.arange(len(features)), results[\"ablated_patches\"])\nwith torch.inference_mode():\n    prob_baseline = model(features).logits.softmax(1)[0, 1].item()  # 1.000\n    prob_without_high_effect = model(features[patches_not_ablated]).logits.softmax(1)[0, 1].item()  # 0.008\n\nprint(f\"Probability of metastasis at baseline: {prob_baseline:0.3f}\")\nprint(f\"Probability of metastasis after removing 1% highest effect patches: {prob_without_high_effect:0.3f}\")\n```\n\nWe can also plot the model outputs as we remove high effect patches, and we hope to see a monotonically decreasing line.\n\n```python\nimport matplotlib.pyplot as plt\nimport numpy as np\n\nmodel_results = results[\"model_outputs\"][:, results[\"optimized_class_index\"]]\nplt.plot(model_results)\nplt.xlabel(\"Number of patches removed\")\nplt.ylabel(\"Probability of metastasis\")\n```\n\n\n# Cite\n\n```bibtex\n@misc{kaczmarzyk2024explainableaicomputationalpathology,\n      title={Explainable AI for computational pathology identifies model limitations and tissue biomarkers},\n      author={Jakub R. Kaczmarzyk and Joel H. Saltz and Peter K. Koo},\n      year={2024},\n      eprint={2409.03080},\n      archivePrefix={arXiv},\n      primaryClass={q-bio.TO},\n      url={https://arxiv.org/abs/2409.03080},\n}\n```\n\n# License\n\nHIPPO code is licensed under the terms of the 3-Clause BSD License, and documentation is published under the terms of the Creative Commons Attribution-NonCommercial-ShareAlike 4.0 International copyright license (CC BY-NC-SA 4.0).\n",
    "bugtrack_url": null,
    "license": "BSD 3-Clause License  Copyright (c) 2024, Jakub Kaczmarzyk  Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met:  1. Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer.  2. Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution.  3. Neither the name of the copyright holder nor the names of its contributors may be used to endorse or promote products derived from this software without specific prior written permission.  THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS \"AS IS\" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.",
    "summary": "HIPPO explainability toolkit for computational pathology.",
    "version": "0.1.0",
    "project_urls": {
        "Bug Tracker": "https://github.com/kaczmarj/HIPPO/issues",
        "Repository": "https://github.com/kaczmarj/HIPPO"
    },
    "split_keywords": [
        "abmil",
        " computational pathology",
        " counterfactuals",
        " deep learning",
        " explainability",
        " multiple instance learning",
        " pathology"
    ],
    "urls": [
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "8c7e34f4ddacb73f6920799449723db01205b1df218ca1f2e51f4bc6a3fc8413",
                "md5": "438267ecfb97b366742a7c1c8682e907",
                "sha256": "c97cf738693f7550cbf4987d797be2e5b70045c44c516cd86edf3098e5c21562"
            },
            "downloads": -1,
            "filename": "hippo_nn-0.1.0-py3-none-any.whl",
            "has_sig": false,
            "md5_digest": "438267ecfb97b366742a7c1c8682e907",
            "packagetype": "bdist_wheel",
            "python_version": "py3",
            "requires_python": ">=3.8",
            "size": 13570,
            "upload_time": "2024-09-06T11:54:04",
            "upload_time_iso_8601": "2024-09-06T11:54:04.124285Z",
            "url": "https://files.pythonhosted.org/packages/8c/7e/34f4ddacb73f6920799449723db01205b1df218ca1f2e51f4bc6a3fc8413/hippo_nn-0.1.0-py3-none-any.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "2567ec16254fc68594e4079f2bf4f06b2461c143e669e3fe3878485fb8ad8d1c",
                "md5": "78220d5e5a517691b996623baaac3d6d",
                "sha256": "3958d69b075ae9e2af9792b72811c6eb71d9ca9b0e6c82c8928f0cf074900526"
            },
            "downloads": -1,
            "filename": "hippo_nn-0.1.0.tar.gz",
            "has_sig": false,
            "md5_digest": "78220d5e5a517691b996623baaac3d6d",
            "packagetype": "sdist",
            "python_version": "source",
            "requires_python": ">=3.8",
            "size": 461915,
            "upload_time": "2024-09-06T11:54:06",
            "upload_time_iso_8601": "2024-09-06T11:54:06.041984Z",
            "url": "https://files.pythonhosted.org/packages/25/67/ec16254fc68594e4079f2bf4f06b2461c143e669e3fe3878485fb8ad8d1c/hippo_nn-0.1.0.tar.gz",
            "yanked": false,
            "yanked_reason": null
        }
    ],
    "upload_time": "2024-09-06 11:54:06",
    "github": true,
    "gitlab": false,
    "bitbucket": false,
    "codeberg": false,
    "github_user": "kaczmarj",
    "github_project": "HIPPO",
    "travis_ci": false,
    "coveralls": false,
    "github_actions": false,
    "lcname": "hippo-nn"
}
        
Elapsed time: 1.61100s