liverquant


Nameliverquant JSON
Version 0.0.4 PyPI version JSON
download
home_page
SummaryQuantitate liver histopathology in whole slide images
upload_time2023-12-07 17:36:30
maintainer
docs_urlNone
author
requires_python>=3.7
license
keywords python geojson opencv liver digital pathology
VCS
bugtrack_url
requirements No requirements were recorded.
Travis-CI No Travis.
coveralls test coverage No coveralls.
            # liverquant
Liverquant is a Python package designed for automated analysis of Whole Slide Images (WSI) to assess Non-alcoholic Fatty Liver Disease (NAFLD) or Metabolic dysfunction Associated Steatotic Liver Disease (MASLD). This toolbox enables the detection of fat globules within image tiles ([Example 1](#example-1)) and is directly applicable to WSIs ([Example 2](#example-2)). In our evaluation, we utilized this toolbox on a [public dataset](#genotype-tissue-expression-gtex-public-dataset) comprising 109 WSIs. Additionally, the toolbox facilitates fibrosis estimation, demonstrated in [Example 3](#example-3) and [Example 4](#example-4). Future plans involve expanding its capabilities to quantify inflammation and ballooning. 

## Contents
- [Installation](#installation)
- [Fat Quantification](#fat-quantification)
  - [detect_fat_globules](#detect_fat_globules)
  - [detect_fat_globules_wsi](#detect_fat_globules_wsi)
  - [Genotype-Tissue Expression (GTEx) public dataset](#genotype-tissue-expression-gtex-public-dataset)
- [Fibrosis Quantification](#fibrosis-quantification)
  - [segment_fibrosis](#segment_fibrosis)
  - [segment_fibrosis_wsi](#segment_fibrosis_wsi)

## Installation
The recommended way to install is via pip:

`pip install liverquant`

## Fat Quantification
In liver pathology, the presence of fat can be indicative of various conditions, such as fatty liver disease (steatosis), which can occur in the context of alcohol abuse, obesity, diabetes, or metabolic syndrome. Hematoxylin and Eosin (H&E) staining is a widely used technique in pathology that provides basic information about tissue architecture and cellular morphology. Under H&E staining, lipids (fats) appear as clear or pale vacuoles within cells. The workflow to identify fat vacuoles is shown below. In brief, white regions are segmented using _Hue-Saturation-Value_ channels. Isolated fat vacuoles are segmented using morphological features. Overlapping fat globules are separated using a watershed segmentation and then identified using the same filters applied to isolated fat vacuoles.

<figure>
  <img src="https://github.com/mfarzi/liverquant/raw/main/example/fat_quant_workflow.jpg" alt="Fat Quantification Workflow" style="width:90%; margin-right:10px;" />
</figure>

> Note: The image tile used in this example is downloaded from [histology page](https://gtexportal.org/home/histologyPage) with the tissue sample ID _GTEX-12584-1526_. 

The library implements two main methods to identify fat globules.

### <code>detect_fat_globules</code>
Retrieve a binary mask for fat vacuoles from input RGB image tile.
- Parameters:
  - img: {numpy.ndarray}: input RGB image tile in range [0, 255]
  - mask: {numpy.ndarray}: binary mask (either 0 or 255) for regions-of-interest [default=None]
  - lowerb: {list: 3}: inclusive lower bound array in HSV-space for color segmentation [default=[0, 0, 200]]
  - upperb: {list: 3}: inclusive upper bound array in HSV-space for color segmentation [default=[180, 25, 255]]
  - overlap: {int}: globules with centre inside overlap region will be excluded to avoid double-counting in whole-slide-image [default=0]
  - resolution: {float}: pixel size in micro-meter [default=1]
  - min_diameter: {float}: minimum diameter of a white blob to be considered as a fat globule [default=5]
  - max_diameter: {float}: maximum diameter of a white blob to be considered as a fat globule [default=100]
- Returns:
  - mask: {numpy.ndarray}: binary mask (either 0 or 255) for identified fat vacuoles

> For color segmentation, you must provide the lower and upper bounds for _Hue-Saturation-Value_ channels. Note that similar to _OpenCV_, Hue has values from 0 to 180, Saturation and Value from 0 to 255. To pick the white color, Saturation between 0 and 25 is filtered out by default. Overlapping fat globules are separated using a watershed segmentation. Fat globules are then identified by filtering blobs based on morphological features including size, solidity, and elongation.

#### Example 1
Here is a short script to demonstrate the utility of `detect_fat_globules`.
<figure>
  <img src="https://github.com/mfarzi/liverquant/raw/main/example/fat_detection.jpg" alt="Fat Detection Example" style="width:70%; margin-right:10px;" />
</figure>

```
import cv2 as cv
from liverquant import detect_fat_globules

# read sample image tile
img = cv.imread('./example/tile01_HE.jpg')
img = cv.cvtColor(img, cv.COLOR_BGR2RGB)

# Detect globules
mask = detect_fat_globules(img, resolution=0.4942)                                           

# Tag globules with green color                                   
img[mask == 255, :] = [5, 255, 5]

# write the tagged image
cv.imwrite('./example/tile01_HE_tagged.jpg', cv.cvtColor(img, cv.COLOR_BGR2RGB))
```

### <code>detect_fat_globules_wsi</code>
Retrieve fat vacuoles in `cv2geojson.GeoContour` format from the whole slide image (WSI).
- Parameters:
  - slide: {openslide}: A handle object to the whole slide image
  - roi: {list: cv2geojson.GeoContours}: regions-of-interest in the WSI frame [default=None]
  - lowerb: {list: 3}: inclusive lower bound array in HSV-space for color segmentation [default=[0, 0, 200]]
  - upperb: {list: 3}: inclusive upper bound array in HSV-space for color segmentation [default=[180, 25, 255]]
  - tile_size: {int}: the tile size to sweep the whole slide image [default=2048]
  - overlap: {int}: globules with centre inside overlap region will be excluded to avoid double-counting in whole-slide-image [default=128]
  - downsample: {int}: downsampling ratio as a power of two [default=2]
  - min_diameter: {float}: minimum diameter of a white blob to be considered as a fat globule [default=5]
  - max_diameter: {float}: maximum diameter of a white blob to be considered as a fat globule [default=100]
  - cores_num: {int}: max number of cores to be used for parallel computation 
- Returns:
  - fat_proportion_area: {float}: the amount of fat in the tissue represented in percent
  - vacoules: {list: cv2geojson.GeoContour}: polygons representing fat vacoules
  - roi: {list: cv2geojson.GeoContour}: same as input roi if provided; otherwise, representing the foreground mask
  - run_time: {float}: run time for the code completion in seconds

> This function divides the input image into overlapping image tiles and applies the `detect_fat_globules` algorithm (refer to [Example 1](#example-1)) to each image tile. `detect_fat_globules_wsi` employs parallel computation, and the entire script should be enclosed within a `if __name__ == '__main__'` block. Liverquant export detected geometrical features in [geojson](https://geojson.org/) format using the [`cv2geojson`](https://github.com/mfarzi/cv2geojson) python package, which can be visualised using dedicated software tools like [QuPath](https://qupath.github.io/). 

#### Example 2
Here is a short script to demonstrate the utility of `detect_fat_globules_wsi`. The WSI used in this example is not provided due to its large size but can be downloaded from [histology page](https://gtexportal.org/home/histologyPage) with the tissue sample ID _GTEX-12584-1526_. The estimated fat proportionate area is 13.8% and the run time was about 214 seconds.

<figure>
  <img src="https://github.com/mfarzi/liverquant/raw/main/example/fat_detection_wsi_qupath_screenshot.png" alt="Visualise Fat globules in WSI using QuPath" style="width:80%; margin-right:10px;" />
</figure>

```
from liverquant import detect_fat_globules_wsi
from cv2geojson import export_annotations
from openslide import OpenSlide

if __name__ == '__main__':
    # open the whole slide image
    slide = OpenSlide('./example/GTEX-12584-1526.svs')

    # detect fat globules
    fat_proportionate_area, globules, roi, run_time = detect_fat_globules_wsi(slide, cores_num=12)

    # export geojson features
    features = []
    for geocontour in roi:
        features.append(geocontour.export_feature(color=(0, 0, 255), label='foreground'))
    for geocontour in globules:
        features.append(geocontour.export_feature(color=(0, 255, 0), label='fat'))
    export_annotations(features, './example/GTEX-12584-1526.geojson')

    # print out results
    print(f'The fat proportionate area is {fat_proportionate_area}%.')
    print(f'The run time is {run_time} seconds.')
```

### Genotype-Tissue Expression (GTEx) public dataset
Our validation process for the steatosis quantification algorithm involved analyzing H&E whole slide images from the Genotype-Tissue Expression (GTEx) public portal, sourced from the diverse collection of histology images obtained from various tissue types in postmortem donors. Out of the 262 liver biopsy cases selected for this study, exhibiting steatosis, Kleiner scores were automatically extracted from pathology notes for 109 cases. This extraction involved scanning pathology notes to assess the steatosis proportionate area (SPA), categorized into four groups: 0 (SPA≤5), 1 (5<SPA≤33), 2 (33<SPA≤66), and 3 (66<SPA). For convenient access, the list of these 109 slides can be downloaded [here]. 

To evaluate the steatosis quantification algorithm, we analysed an external public dataset of H&E whole slide images from the Genotype-Tissue Expression ([GTEx](https://gtexportal.org/home/histologyPage)) public portal (Broad Institute, Cambridge, MA, USA). The GTEx tissue image repository includes a diverse collection of histology images derived from various tissue types obtained from postmortem donors. Out of the 262 liver biopsy cases selected for this study, exhibiting steatosis, Kleiner scores were automatically extracted from pathology notes for 109 cases. To achieve this, pathology notes were scanned for the steatosis proportionate area (SPA) and then categorised into four groups: 0 (SPA≤5), 1 (5<SPA≤33), 2 (33<SPA≤66), and 3 (66<SPA). For convenient access, the list of these 109 slides can be downloaded [here](https://github.com/mfarzi/liverquant/blob/main/example/gtex_liver_steatosis_dataset.csv). The correlation between estimated SPA at 20x magnification and extracted Kleiner scores is visually represented in the box-and-whisker plot below, showing a moderate Spearman rank correlation (ρ=0.75) within this subset of N=109 cases.

<p align="center">
  <img src="https://github.com/mfarzi/liverquant/raw/main/example/box_plot_gtxportal_steatosis_20x.png" alt="Box-and-Whisker plot for GTex liver dataset for steatosis quantification" style="width:50%; margin-right:10px;" />
</p>

You can access and download the slides using the following Python script:
```
import requests
from pathlib import Path
import pandas as pd

# define the local working directory 
root_path = Path('.\\Example') 
path_to_dataset = root_path / 'gtex_liver_steatosis_dataset.csv'
df = pd.read_csv(path_to_dataset)


def download_file(url, destination):
    try:
        response = requests.get(url, stream=True)
        response.raise_for_status()

        with open(destination, 'wb') as file:
            for chunk in response.iter_content(chunk_size=8192):
                if chunk:
                    file.write(chunk)

        print(f"File downloaded to {destination}")
    except requests.exceptions.RequestException as e:
        print(f"An error occurred: {e}")


if __name__ == "__main__":
    # create a folder to store whole slide images
    path_to_slides = root_path / 'slides'
    if not path_to_slides.is_dir():
        path_to_slides.mkdir()

    # download the slides
    for i in range(df.shape[0]):
        slide_name = df['Slide ID'][i]
        file_url = "https://brd.nci.nih.gov/brd/imagedownload/{}".format(slide_name)
        destination_file = path_to_slides / '{}.svs'.format(slide_name)

        if not destination_file.is_file():
            print(f'downloading {file_url}...')
            download_file(file_url, destination_file)

```

Upon downloading the slides, utilize the script below to compute the steatosis proportionate area (SPA).
```
from pathlib import Path
import pandas as pd
from liverquant import detect_fat_globules_wsi
import numpy as np
from openslide import OpenSlide

root_path = Path('.\\Example') # local working directory
path_to_slides = root_path / 'slides'
path_to_dataset = root_path / 'gtex_liver_steatosis_dataset.csv'

if __name__ == '__main__':
    df = pd.read_csv(path_to_dataset)
    for index in range(df.shape[0]):
        slide_id = df['Slide ID'][index]
        slide_filename = path_to_slides / '{}.svs'.format(slide_id)

        slide = OpenSlide(slide_filename)
        spa, contours, roi, run_time = detect_fat_globules_wsi(slide,
                                                               upperb=[180, 25, 255],
                                                               lowerb=[0, 0, 200],
                                                               tile_size=2048,
                                                               overlap=128,
                                                               downsample=1,
                                                               min_diameter=5,
                                                               max_diameter=100,
                                                               cores_num=12)

        # store results
        df.at[index, 'SPA'] = np.round(spa, 2)
        df.at[index, 'Runtime'] = np.round(run_time, 3)
        print(f'slide {slide_id}--- fpa: {spa}  run time:{run_time}')
    df.to_csv(path_to_dataset, index=False)

```

## Fibrosis Quantification
Liver fibrosis is a progressive condition characterized by the excessive accumulation of extracellular matrix components, particularly collagen, within the liver tissue. It is typically a consequence of chronic liver injury caused by various factors such as viral hepatitis, alcohol abuse, metabolic disorders, or autoimmune diseases. The excessive collagen deposition disrupts the normal liver architecture and impairs its function over time.

To evaluate the extent of liver fibrosis, histological staining techniques are commonly employed. Picrosirius red (PSR), Masson's Trichrome (MTC), and Van Gieson (VG) staining are three widely used methods to visualize and quantify the collagen content within liver tissue. These techniques allow for the identification of collagen fibers and provide valuable information about the severity and distribution of fibrosis.

1- Picrosirius red (PSR) staining: PSR staining specifically highlights collagen fibers and helps differentiate between mature (thick, red-orange staining) and immature (thin, green staining) collagen. This staining technique allows for the assessment of collagen distribution, fiber thickness, and the presence of newly formed collagen.

2- Masson's Trichrome (MTC) staining: MTC staining is a widely used technique to visualize collagen fibers in tissues. It involves staining collagen fibers blue or green, nuclei dark blue or black, and cytoplasm and muscle fibers red. Collagen fibers appear as blue-green structures, allowing for the identification and quantification of fibrosis. MTC staining provides information about the extent of collagen deposition and can help evaluate fibrosis severity.

3- Van Gieson (VG) staining: VG staining is another method widely used to assess collagen deposition in tissues. It stains collagen fibers red or pink, while other tissue components such as muscle fibers, cytoplasm, and nuclei are stained yellow or brown. This staining technique allows for the identification and quantification of collagen within the liver tissue, enabling the evaluation of fibrosis severity.

The library implements two main methods to identify collagens using colour segmentation in both individual image tiles and the whole slide image.

### <code>segment_fibrosis</code>
Retrieve a binary mask for segmented regions based on their colour profile in the HSV space.
- Parameters:
  - img: {numpy.ndarray}: input RGB image tile in range [0, 255]
  - mask: {numpy.ndarray}: binary mask (either 0 or 255) for regions-of-interest [default=None]
  - stain: {string}: VG or PSR or MTC
  - lowerb: {list: 3}: inclusive lower bound array in HSV-space for color segmentation [default=None]
  - upperb: {list: 3}: inclusive upper bound array in HSV-space for color segmentation [default=None]
  - mixing_matrix: {numpy.ndarray}: the proportion of each wavelength absorbed from red, green, and blue channels [default=None]
  - ref_mixing_matrix: {numpy.ndarray}: reference mixing matrix to normalise input image
  - scale: {list: 3}: global scaling factor for stain normalisation
  - hole_size: {float}: remove holes smaller than hole_size; if zero, all holes are reserved. If -1, all holes
                        will be removed. [default=0]
  - blob_size: {float}: remove blobs smaller than blob_size; if zero, all blobs are reserved.
  - resolution: {float}: pixel size in micro-meter [default=1]
  
- Returns:
  - cpa: {float}: Collagen Proportionate Area (CPA) in percent
  - collagen_mask: {numpy.ndarray}: binary mask (either 0 or 255) for segmented regions
  - roi: {numpy.ndarray}: binary mask (either 0 or 255) for the regions of interest

> Note that similar to _OpenCV_, Hue has values from 0 to 180, Saturation and Value from 0 to 255.

#### Example 3
Here is a short script to demonstrate the utility of `segment_fibrosis`. An image tile stained with PSR is provided in the example folder. To segment the collagen shown in red, Hue between 0 and 10 or 170-180 should be filtered out. We have combined the two ranges into one effective range [-10, 10] in the sample code below. Since Hue is inherently periodic, negative Hue can be interpreted as positive integers by adding 180. Note that similar to _OpenCV_, Hue has values from 0 to 180.

<figure>
  <img src="https://github.com/mfarzi/liverquant/raw/main/example/collagen_segmentation.jpg" alt="Collage Segmentation Example" style="width:80%; margin-right:10px;" />
</figure>

```
import cv2 as cv
from liverquant import segment_fibrosis

# read sample image tile
img = cv.imread('./example/tile02_PSR.jpg')
img = cv.cvtColor(img, cv.COLOR_BGR2RGB)

cpa, mask, roi = segment_fibrosis(img,
                                  lowerb=[-10, 50, 100],
                                  upperb=[10, 255, 255],
                                  resolution=1.0124)

print(f'CPA = {cpa}%')

# write the tagged image
cv.imwrite('./example/tile02_PSR_mask.jpg', mask)
```

### <code>segment_fibrosis_wsi</code>
Retrieve segmented regions in `cv2geojson.GeoContour` format based on their colour profile in the HSV space from the whole slide image (WSI).
- Parameters:
  - img: {numpy.ndarray}: input RGB image tile in range [0, 255]
  - stain: {string}: VG or PSR or MTC
  - roi: {list: cv2geojson.GeoContours}: regions-of-interest in the WSI frame [default=None]
  - lowerb: {list: 3}: inclusive lower bound array in HSV-space for color segmentation [default=None]
  - upperb: {list: 3}: inclusive upper bound array in HSV-space for color segmentation [default=None]
  - mixing_matrix: {numpy.ndarray}: the proportion of each wavelength absorbed from red, green, and blue channels [default=None]
  - ref_mixing_matrix: {numpy.ndarray}: reference mixing matrix to normalise input image
  - scale: {list: 3}: global scaling factor for stain normalisation
  
  - hole_size: {float}: remove holes smaller than hole_size; if zero, all holes will be reserved. If -1, all holes
                        will be removed. [default=0]
  - blob_size: {float}: remove blobs smaller than blob_size; if zero, all blobs are reserved [default=0]
  - tile_size: {int}: the tile size to sweep the whole slide image [default=2048]
  - downsample: {int}: downsampling ratio as a power of two [default=2]
  - cores_num: {int}: max number of cores to be used for parallel computation [default=4]
  
- Returns:
  - cpa: {float}: Collagen Proportionate Area (CPA) in percent
  - geocontours: {list: cv2geojson.GeoContour}: polygons representing segmented fibrotic regions
  - roi: {numpy.ndarray}: binary mask (either 0 or 255) for the regions of interest

> Note that similar to _OpenCV_, Hue has values from 0 to 180, Saturation and Value from 0 to 255. This function divides the input image into image tiles and applies the `segment_by_color` algorithm (refer to [Example 3](#example-3)) to each image tile. `segment_by_color_wsi` employs parallel computation, and the entire script should be enclosed within a `if __name__ == '__main__'` block. Liverquant export detected geometrical features in [geojson](https://geojson.org/) format using the [`cv2geojson`](https://github.com/mfarzi/cv2geojson) python package, which can be visualised using dedicated software tools like [QuPath](https://qupath.github.io/). 

#### Example 4
Here is a short script to demonstrate the utility of `segment_fibrosis_wsi`. To segment the collagen shown in red, Hue between 0 and 10 or 170-180 should be filtered out. Since Hue is inherently periodic, negative Hue can be interpreted as positive integers by adding 180. We have combined the two ranges into one effective range [-10, 10]in the sample code below.

```
from liverquant import segment_fibrosis_wsi
from cv2geojson import export_annotations
from openslide import OpenSlide

if __name__ == '__main__':
    # open the whole slide image
    slide = OpenSlide('./example/PSR_slide.svs')

    # Segment collagen using Hue-Saturation-Value channels
    cpa, geocontours, roi = segment_fibrosis_wsi(slide,
                                                 stain='PSR',
                                                 lowerb=[-10, 50, 100],
                                                 upperb=[10, 255, 255],
                                                 downsample=16,
                                                 tile_size=4096,
                                                 cores_num=12)

    # print out results
    print(f'The collagen proportionate area is {cpa}%.')

    # export geojson features
    features = []
    for geocontour in roi:
        features.append(geocontour.export_feature(color=(255, 0, 0), label='foreground'))
    for geocontour in geocontours:
        features.append(geocontour.export_feature(color=(0, 0, 255), label='fibrosis'))
    export_annotations(features, './example/PSR_slide.geojson')

```

            

Raw data

            {
    "_id": null,
    "home_page": "",
    "name": "liverquant",
    "maintainer": "",
    "docs_url": null,
    "requires_python": ">=3.7",
    "maintainer_email": "",
    "keywords": "python,geojson,opencv,liver,digital pathology",
    "author": "",
    "author_email": "Mohsen Farzi <mhnfarzi@gmail.com>",
    "download_url": "https://files.pythonhosted.org/packages/a6/e2/5c78d4573c1a6d579a0582b4020a17f9b56fccc2a06af43eaa73f2977571/liverquant-0.0.4.tar.gz",
    "platform": null,
    "description": "# liverquant\r\nLiverquant is a Python package designed for automated analysis of Whole Slide Images (WSI) to assess Non-alcoholic Fatty Liver Disease (NAFLD) or Metabolic dysfunction Associated Steatotic Liver Disease (MASLD). This toolbox enables the detection of fat globules within image tiles ([Example 1](#example-1)) and is directly applicable to WSIs ([Example 2](#example-2)). In our evaluation, we utilized this toolbox on a [public dataset](#genotype-tissue-expression-gtex-public-dataset) comprising 109 WSIs. Additionally, the toolbox facilitates fibrosis estimation, demonstrated in [Example 3](#example-3) and [Example 4](#example-4). Future plans involve expanding its capabilities to quantify inflammation and ballooning. \r\n\r\n## Contents\r\n- [Installation](#installation)\r\n- [Fat Quantification](#fat-quantification)\r\n  - [detect_fat_globules](#detect_fat_globules)\r\n  - [detect_fat_globules_wsi](#detect_fat_globules_wsi)\r\n  - [Genotype-Tissue Expression (GTEx) public dataset](#genotype-tissue-expression-gtex-public-dataset)\r\n- [Fibrosis Quantification](#fibrosis-quantification)\r\n  - [segment_fibrosis](#segment_fibrosis)\r\n  - [segment_fibrosis_wsi](#segment_fibrosis_wsi)\r\n\r\n## Installation\r\nThe recommended way to install is via pip:\r\n\r\n`pip install liverquant`\r\n\r\n## Fat Quantification\r\nIn liver pathology, the presence of fat can be indicative of various conditions, such as fatty liver disease (steatosis), which can occur in the context of alcohol abuse, obesity, diabetes, or metabolic syndrome. Hematoxylin and Eosin (H&E) staining is a widely used technique in pathology that provides basic information about tissue architecture and cellular morphology. Under H&E staining, lipids (fats) appear as clear or pale vacuoles within cells. The workflow to identify fat vacuoles is shown below. In brief, white regions are segmented using _Hue-Saturation-Value_ channels. Isolated fat vacuoles are segmented using morphological features. Overlapping fat globules are separated using a watershed segmentation and then identified using the same filters applied to isolated fat vacuoles.\r\n\r\n<figure>\r\n  <img src=\"https://github.com/mfarzi/liverquant/raw/main/example/fat_quant_workflow.jpg\" alt=\"Fat Quantification Workflow\" style=\"width:90%; margin-right:10px;\" />\r\n</figure>\r\n\r\n> Note: The image tile used in this example is downloaded from [histology page](https://gtexportal.org/home/histologyPage) with the tissue sample ID _GTEX-12584-1526_. \r\n\r\nThe library implements two main methods to identify fat globules.\r\n\r\n### <code>detect_fat_globules</code>\r\nRetrieve a binary mask for fat vacuoles from input RGB image tile.\r\n- Parameters:\r\n  - img: {numpy.ndarray}: input RGB image tile in range [0, 255]\r\n  - mask: {numpy.ndarray}: binary mask (either 0 or 255) for regions-of-interest [default=None]\r\n  - lowerb: {list: 3}: inclusive lower bound array in HSV-space for color segmentation [default=[0, 0, 200]]\r\n  - upperb: {list: 3}: inclusive upper bound array in HSV-space for color segmentation [default=[180, 25, 255]]\r\n  - overlap: {int}: globules with centre inside overlap region will be excluded to avoid double-counting in whole-slide-image [default=0]\r\n  - resolution: {float}: pixel size in micro-meter [default=1]\r\n  - min_diameter: {float}: minimum diameter of a white blob to be considered as a fat globule [default=5]\r\n  - max_diameter: {float}: maximum diameter of a white blob to be considered as a fat globule [default=100]\r\n- Returns:\r\n  - mask: {numpy.ndarray}: binary mask (either 0 or 255) for identified fat vacuoles\r\n\r\n> For color segmentation, you must provide the lower and upper bounds for _Hue-Saturation-Value_ channels. Note that similar to _OpenCV_, Hue has values from 0 to 180, Saturation and Value from 0 to 255. To pick the white color, Saturation between 0 and 25 is filtered out by default. Overlapping fat globules are separated using a watershed segmentation. Fat globules are then identified by filtering blobs based on morphological features including size, solidity, and elongation.\r\n\r\n#### Example 1\r\nHere is a short script to demonstrate the utility of `detect_fat_globules`.\r\n<figure>\r\n  <img src=\"https://github.com/mfarzi/liverquant/raw/main/example/fat_detection.jpg\" alt=\"Fat Detection Example\" style=\"width:70%; margin-right:10px;\" />\r\n</figure>\r\n\r\n```\r\nimport cv2 as cv\r\nfrom liverquant import detect_fat_globules\r\n\r\n# read sample image tile\r\nimg = cv.imread('./example/tile01_HE.jpg')\r\nimg = cv.cvtColor(img, cv.COLOR_BGR2RGB)\r\n\r\n# Detect globules\r\nmask = detect_fat_globules(img, resolution=0.4942)                                           \r\n\r\n# Tag globules with green color                                   \r\nimg[mask == 255, :] = [5, 255, 5]\r\n\r\n# write the tagged image\r\ncv.imwrite('./example/tile01_HE_tagged.jpg', cv.cvtColor(img, cv.COLOR_BGR2RGB))\r\n```\r\n\r\n### <code>detect_fat_globules_wsi</code>\r\nRetrieve fat vacuoles in `cv2geojson.GeoContour` format from the whole slide image (WSI).\r\n- Parameters:\r\n  - slide: {openslide}: A handle object to the whole slide image\r\n  - roi: {list: cv2geojson.GeoContours}: regions-of-interest in the WSI frame [default=None]\r\n  - lowerb: {list: 3}: inclusive lower bound array in HSV-space for color segmentation [default=[0, 0, 200]]\r\n  - upperb: {list: 3}: inclusive upper bound array in HSV-space for color segmentation [default=[180, 25, 255]]\r\n  - tile_size: {int}: the tile size to sweep the whole slide image [default=2048]\r\n  - overlap: {int}: globules with centre inside overlap region will be excluded to avoid double-counting in whole-slide-image [default=128]\r\n  - downsample: {int}: downsampling ratio as a power of two [default=2]\r\n  - min_diameter: {float}: minimum diameter of a white blob to be considered as a fat globule [default=5]\r\n  - max_diameter: {float}: maximum diameter of a white blob to be considered as a fat globule [default=100]\r\n  - cores_num: {int}: max number of cores to be used for parallel computation \r\n- Returns:\r\n  - fat_proportion_area: {float}: the amount of fat in the tissue represented in percent\r\n  - vacoules: {list: cv2geojson.GeoContour}: polygons representing fat vacoules\r\n  - roi: {list: cv2geojson.GeoContour}: same as input roi if provided; otherwise, representing the foreground mask\r\n  - run_time: {float}: run time for the code completion in seconds\r\n\r\n> This function divides the input image into overlapping image tiles and applies the `detect_fat_globules` algorithm (refer to [Example 1](#example-1)) to each image tile. `detect_fat_globules_wsi` employs parallel computation, and the entire script should be enclosed within a `if __name__ == '__main__'` block. Liverquant export detected geometrical features in [geojson](https://geojson.org/) format using the [`cv2geojson`](https://github.com/mfarzi/cv2geojson) python package, which can be visualised using dedicated software tools like [QuPath](https://qupath.github.io/). \r\n\r\n#### Example 2\r\nHere is a short script to demonstrate the utility of `detect_fat_globules_wsi`. The WSI used in this example is not provided due to its large size but can be downloaded from [histology page](https://gtexportal.org/home/histologyPage) with the tissue sample ID _GTEX-12584-1526_. The estimated fat proportionate area is 13.8% and the run time was about 214 seconds.\r\n\r\n<figure>\r\n  <img src=\"https://github.com/mfarzi/liverquant/raw/main/example/fat_detection_wsi_qupath_screenshot.png\" alt=\"Visualise Fat globules in WSI using QuPath\" style=\"width:80%; margin-right:10px;\" />\r\n</figure>\r\n\r\n```\r\nfrom liverquant import detect_fat_globules_wsi\r\nfrom cv2geojson import export_annotations\r\nfrom openslide import OpenSlide\r\n\r\nif __name__ == '__main__':\r\n    # open the whole slide image\r\n    slide = OpenSlide('./example/GTEX-12584-1526.svs')\r\n\r\n    # detect fat globules\r\n    fat_proportionate_area, globules, roi, run_time = detect_fat_globules_wsi(slide, cores_num=12)\r\n\r\n    # export geojson features\r\n    features = []\r\n    for geocontour in roi:\r\n        features.append(geocontour.export_feature(color=(0, 0, 255), label='foreground'))\r\n    for geocontour in globules:\r\n        features.append(geocontour.export_feature(color=(0, 255, 0), label='fat'))\r\n    export_annotations(features, './example/GTEX-12584-1526.geojson')\r\n\r\n    # print out results\r\n    print(f'The fat proportionate area is {fat_proportionate_area}%.')\r\n    print(f'The run time is {run_time} seconds.')\r\n```\r\n\r\n### Genotype-Tissue Expression (GTEx) public dataset\r\nOur validation process for the steatosis quantification algorithm involved analyzing H&E whole slide images from the Genotype-Tissue Expression (GTEx) public portal, sourced from the diverse collection of histology images obtained from various tissue types in postmortem donors. Out of the 262 liver biopsy cases selected for this study, exhibiting steatosis, Kleiner scores were automatically extracted from pathology notes for 109 cases. This extraction involved scanning pathology notes to assess the steatosis proportionate area (SPA), categorized into four groups: 0 (SPA\u22645), 1 (5<SPA\u226433), 2 (33<SPA\u226466), and 3 (66<SPA). For convenient access, the list of these 109 slides can be downloaded [here]. \r\n\r\nTo evaluate the steatosis quantification algorithm, we analysed an external public dataset of H&E whole slide images from the Genotype-Tissue Expression ([GTEx](https://gtexportal.org/home/histologyPage)) public portal (Broad Institute, Cambridge, MA, USA). The GTEx tissue image repository includes a diverse collection of histology images derived from various tissue types obtained from postmortem donors. Out of the 262 liver biopsy cases selected for this study, exhibiting steatosis, Kleiner scores were automatically extracted from pathology notes for 109 cases. To achieve this, pathology notes were scanned for the steatosis proportionate area (SPA) and then categorised into four groups: 0 (SPA\u22645), 1 (5<SPA\u226433), 2 (33<SPA\u226466), and 3 (66<SPA). For convenient access, the list of these 109 slides can be downloaded [here](https://github.com/mfarzi/liverquant/blob/main/example/gtex_liver_steatosis_dataset.csv). The correlation between estimated SPA at 20x magnification and extracted Kleiner scores is visually represented in the box-and-whisker plot below, showing a moderate Spearman rank correlation (\u03c1=0.75) within this subset of N=109 cases.\r\n\r\n<p align=\"center\">\r\n  <img src=\"https://github.com/mfarzi/liverquant/raw/main/example/box_plot_gtxportal_steatosis_20x.png\" alt=\"Box-and-Whisker plot for GTex liver dataset for steatosis quantification\" style=\"width:50%; margin-right:10px;\" />\r\n</p>\r\n\r\nYou can access and download the slides using the following Python script:\r\n```\r\nimport requests\r\nfrom pathlib import Path\r\nimport pandas as pd\r\n\r\n# define the local working directory \r\nroot_path = Path('.\\\\Example') \r\npath_to_dataset = root_path / 'gtex_liver_steatosis_dataset.csv'\r\ndf = pd.read_csv(path_to_dataset)\r\n\r\n\r\ndef download_file(url, destination):\r\n    try:\r\n        response = requests.get(url, stream=True)\r\n        response.raise_for_status()\r\n\r\n        with open(destination, 'wb') as file:\r\n            for chunk in response.iter_content(chunk_size=8192):\r\n                if chunk:\r\n                    file.write(chunk)\r\n\r\n        print(f\"File downloaded to {destination}\")\r\n    except requests.exceptions.RequestException as e:\r\n        print(f\"An error occurred: {e}\")\r\n\r\n\r\nif __name__ == \"__main__\":\r\n    # create a folder to store whole slide images\r\n    path_to_slides = root_path / 'slides'\r\n    if not path_to_slides.is_dir():\r\n        path_to_slides.mkdir()\r\n\r\n    # download the slides\r\n    for i in range(df.shape[0]):\r\n        slide_name = df['Slide ID'][i]\r\n        file_url = \"https://brd.nci.nih.gov/brd/imagedownload/{}\".format(slide_name)\r\n        destination_file = path_to_slides / '{}.svs'.format(slide_name)\r\n\r\n        if not destination_file.is_file():\r\n            print(f'downloading {file_url}...')\r\n            download_file(file_url, destination_file)\r\n\r\n```\r\n\r\nUpon downloading the slides, utilize the script below to compute the steatosis proportionate area (SPA).\r\n```\r\nfrom pathlib import Path\r\nimport pandas as pd\r\nfrom liverquant import detect_fat_globules_wsi\r\nimport numpy as np\r\nfrom openslide import OpenSlide\r\n\r\nroot_path = Path('.\\\\Example') # local working directory\r\npath_to_slides = root_path / 'slides'\r\npath_to_dataset = root_path / 'gtex_liver_steatosis_dataset.csv'\r\n\r\nif __name__ == '__main__':\r\n    df = pd.read_csv(path_to_dataset)\r\n    for index in range(df.shape[0]):\r\n        slide_id = df['Slide ID'][index]\r\n        slide_filename = path_to_slides / '{}.svs'.format(slide_id)\r\n\r\n        slide = OpenSlide(slide_filename)\r\n        spa, contours, roi, run_time = detect_fat_globules_wsi(slide,\r\n                                                               upperb=[180, 25, 255],\r\n                                                               lowerb=[0, 0, 200],\r\n                                                               tile_size=2048,\r\n                                                               overlap=128,\r\n                                                               downsample=1,\r\n                                                               min_diameter=5,\r\n                                                               max_diameter=100,\r\n                                                               cores_num=12)\r\n\r\n        # store results\r\n        df.at[index, 'SPA'] = np.round(spa, 2)\r\n        df.at[index, 'Runtime'] = np.round(run_time, 3)\r\n        print(f'slide {slide_id}--- fpa: {spa}  run time:{run_time}')\r\n    df.to_csv(path_to_dataset, index=False)\r\n\r\n```\r\n\r\n## Fibrosis Quantification\r\nLiver fibrosis is a progressive condition characterized by the excessive accumulation of extracellular matrix components, particularly collagen, within the liver tissue. It is typically a consequence of chronic liver injury caused by various factors such as viral hepatitis, alcohol abuse, metabolic disorders, or autoimmune diseases. The excessive collagen deposition disrupts the normal liver architecture and impairs its function over time.\r\n\r\nTo evaluate the extent of liver fibrosis, histological staining techniques are commonly employed. Picrosirius red (PSR), Masson's Trichrome (MTC), and Van Gieson (VG) staining are three widely used methods to visualize and quantify the collagen content within liver tissue. These techniques allow for the identification of collagen fibers and provide valuable information about the severity and distribution of fibrosis.\r\n\r\n1- Picrosirius red (PSR) staining: PSR staining specifically highlights collagen fibers and helps differentiate between mature (thick, red-orange staining) and immature (thin, green staining) collagen. This staining technique allows for the assessment of collagen distribution, fiber thickness, and the presence of newly formed collagen.\r\n\r\n2- Masson's Trichrome (MTC) staining: MTC staining is a widely used technique to visualize collagen fibers in tissues. It involves staining collagen fibers blue or green, nuclei dark blue or black, and cytoplasm and muscle fibers red. Collagen fibers appear as blue-green structures, allowing for the identification and quantification of fibrosis. MTC staining provides information about the extent of collagen deposition and can help evaluate fibrosis severity.\r\n\r\n3- Van Gieson (VG) staining: VG staining is another method widely used to assess collagen deposition in tissues. It stains collagen fibers red or pink, while other tissue components such as muscle fibers, cytoplasm, and nuclei are stained yellow or brown. This staining technique allows for the identification and quantification of collagen within the liver tissue, enabling the evaluation of fibrosis severity.\r\n\r\nThe library implements two main methods to identify collagens using colour segmentation in both individual image tiles and the whole slide image.\r\n\r\n### <code>segment_fibrosis</code>\r\nRetrieve a binary mask for segmented regions based on their colour profile in the HSV space.\r\n- Parameters:\r\n  - img: {numpy.ndarray}: input RGB image tile in range [0, 255]\r\n  - mask: {numpy.ndarray}: binary mask (either 0 or 255) for regions-of-interest [default=None]\r\n  - stain: {string}: VG or PSR or MTC\r\n  - lowerb: {list: 3}: inclusive lower bound array in HSV-space for color segmentation [default=None]\r\n  - upperb: {list: 3}: inclusive upper bound array in HSV-space for color segmentation [default=None]\r\n  - mixing_matrix: {numpy.ndarray}: the proportion of each wavelength absorbed from red, green, and blue channels [default=None]\r\n  - ref_mixing_matrix: {numpy.ndarray}: reference mixing matrix to normalise input image\r\n  - scale: {list: 3}: global scaling factor for stain normalisation\r\n  - hole_size: {float}: remove holes smaller than hole_size; if zero, all holes are reserved. If -1, all holes\r\n                        will be removed. [default=0]\r\n  - blob_size: {float}: remove blobs smaller than blob_size; if zero, all blobs are reserved.\r\n  - resolution: {float}: pixel size in micro-meter [default=1]\r\n  \r\n- Returns:\r\n  - cpa: {float}: Collagen Proportionate Area (CPA) in percent\r\n  - collagen_mask: {numpy.ndarray}: binary mask (either 0 or 255) for segmented regions\r\n  - roi: {numpy.ndarray}: binary mask (either 0 or 255) for the regions of interest\r\n\r\n> Note that similar to _OpenCV_, Hue has values from 0 to 180, Saturation and Value from 0 to 255.\r\n\r\n#### Example 3\r\nHere is a short script to demonstrate the utility of `segment_fibrosis`. An image tile stained with PSR is provided in the example folder. To segment the collagen shown in red, Hue between 0 and 10 or 170-180 should be filtered out. We have combined the two ranges into one effective range [-10, 10] in the sample code below. Since Hue is inherently periodic, negative Hue can be interpreted as positive integers by adding 180. Note that similar to _OpenCV_, Hue has values from 0 to 180.\r\n\r\n<figure>\r\n  <img src=\"https://github.com/mfarzi/liverquant/raw/main/example/collagen_segmentation.jpg\" alt=\"Collage Segmentation Example\" style=\"width:80%; margin-right:10px;\" />\r\n</figure>\r\n\r\n```\r\nimport cv2 as cv\r\nfrom liverquant import segment_fibrosis\r\n\r\n# read sample image tile\r\nimg = cv.imread('./example/tile02_PSR.jpg')\r\nimg = cv.cvtColor(img, cv.COLOR_BGR2RGB)\r\n\r\ncpa, mask, roi = segment_fibrosis(img,\r\n                                  lowerb=[-10, 50, 100],\r\n                                  upperb=[10, 255, 255],\r\n                                  resolution=1.0124)\r\n\r\nprint(f'CPA = {cpa}%')\r\n\r\n# write the tagged image\r\ncv.imwrite('./example/tile02_PSR_mask.jpg', mask)\r\n```\r\n\r\n### <code>segment_fibrosis_wsi</code>\r\nRetrieve segmented regions in `cv2geojson.GeoContour` format based on their colour profile in the HSV space from the whole slide image (WSI).\r\n- Parameters:\r\n  - img: {numpy.ndarray}: input RGB image tile in range [0, 255]\r\n  - stain: {string}: VG or PSR or MTC\r\n  - roi: {list: cv2geojson.GeoContours}: regions-of-interest in the WSI frame [default=None]\r\n  - lowerb: {list: 3}: inclusive lower bound array in HSV-space for color segmentation [default=None]\r\n  - upperb: {list: 3}: inclusive upper bound array in HSV-space for color segmentation [default=None]\r\n  - mixing_matrix: {numpy.ndarray}: the proportion of each wavelength absorbed from red, green, and blue channels [default=None]\r\n  - ref_mixing_matrix: {numpy.ndarray}: reference mixing matrix to normalise input image\r\n  - scale: {list: 3}: global scaling factor for stain normalisation\r\n  \r\n  - hole_size: {float}: remove holes smaller than hole_size; if zero, all holes will be reserved. If -1, all holes\r\n                        will be removed. [default=0]\r\n  - blob_size: {float}: remove blobs smaller than blob_size; if zero, all blobs are reserved [default=0]\r\n  - tile_size: {int}: the tile size to sweep the whole slide image [default=2048]\r\n  - downsample: {int}: downsampling ratio as a power of two [default=2]\r\n  - cores_num: {int}: max number of cores to be used for parallel computation [default=4]\r\n  \r\n- Returns:\r\n  - cpa: {float}: Collagen Proportionate Area (CPA) in percent\r\n  - geocontours: {list: cv2geojson.GeoContour}: polygons representing segmented fibrotic regions\r\n  - roi: {numpy.ndarray}: binary mask (either 0 or 255) for the regions of interest\r\n\r\n> Note that similar to _OpenCV_, Hue has values from 0 to 180, Saturation and Value from 0 to 255. This function divides the input image into image tiles and applies the `segment_by_color` algorithm (refer to [Example 3](#example-3)) to each image tile. `segment_by_color_wsi` employs parallel computation, and the entire script should be enclosed within a `if __name__ == '__main__'` block. Liverquant export detected geometrical features in [geojson](https://geojson.org/) format using the [`cv2geojson`](https://github.com/mfarzi/cv2geojson) python package, which can be visualised using dedicated software tools like [QuPath](https://qupath.github.io/). \r\n\r\n#### Example 4\r\nHere is a short script to demonstrate the utility of `segment_fibrosis_wsi`. To segment the collagen shown in red, Hue between 0 and 10 or 170-180 should be filtered out. Since Hue is inherently periodic, negative Hue can be interpreted as positive integers by adding 180. We have combined the two ranges into one effective range [-10, 10]in the sample code below.\r\n\r\n```\r\nfrom liverquant import segment_fibrosis_wsi\r\nfrom cv2geojson import export_annotations\r\nfrom openslide import OpenSlide\r\n\r\nif __name__ == '__main__':\r\n    # open the whole slide image\r\n    slide = OpenSlide('./example/PSR_slide.svs')\r\n\r\n    # Segment collagen using Hue-Saturation-Value channels\r\n    cpa, geocontours, roi = segment_fibrosis_wsi(slide,\r\n                                                 stain='PSR',\r\n                                                 lowerb=[-10, 50, 100],\r\n                                                 upperb=[10, 255, 255],\r\n                                                 downsample=16,\r\n                                                 tile_size=4096,\r\n                                                 cores_num=12)\r\n\r\n    # print out results\r\n    print(f'The collagen proportionate area is {cpa}%.')\r\n\r\n    # export geojson features\r\n    features = []\r\n    for geocontour in roi:\r\n        features.append(geocontour.export_feature(color=(255, 0, 0), label='foreground'))\r\n    for geocontour in geocontours:\r\n        features.append(geocontour.export_feature(color=(0, 0, 255), label='fibrosis'))\r\n    export_annotations(features, './example/PSR_slide.geojson')\r\n\r\n```\r\n",
    "bugtrack_url": null,
    "license": "",
    "summary": "Quantitate liver histopathology in whole slide images",
    "version": "0.0.4",
    "project_urls": {
        "Bug Tracker": "https://github.com/mfarzi/liverquant/issues",
        "Homepage": "https://github.com/mfarzi/liverquant"
    },
    "split_keywords": [
        "python",
        "geojson",
        "opencv",
        "liver",
        "digital pathology"
    ],
    "urls": [
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "c50dc886a9ac3636422a2af8484ebb5f812954e16a840d28bbe71265b1ce7c7d",
                "md5": "404bb94c404d853f5a4387bb0021799a",
                "sha256": "b01175b4ba2495635350b0e49c286627f3b4e85359d4b8ed6102b233c3a90abd"
            },
            "downloads": -1,
            "filename": "liverquant-0.0.4-py3-none-any.whl",
            "has_sig": false,
            "md5_digest": "404bb94c404d853f5a4387bb0021799a",
            "packagetype": "bdist_wheel",
            "python_version": "py3",
            "requires_python": ">=3.7",
            "size": 22132,
            "upload_time": "2023-12-07T17:36:28",
            "upload_time_iso_8601": "2023-12-07T17:36:28.310894Z",
            "url": "https://files.pythonhosted.org/packages/c5/0d/c886a9ac3636422a2af8484ebb5f812954e16a840d28bbe71265b1ce7c7d/liverquant-0.0.4-py3-none-any.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "a6e25c78d4573c1a6d579a0582b4020a17f9b56fccc2a06af43eaa73f2977571",
                "md5": "b729591e2007ed2bbae12fd47605946a",
                "sha256": "b6c0d4ca57c23010e014b1ce99cf9a447155d4d1f92d33ed34a8fb7c31546204"
            },
            "downloads": -1,
            "filename": "liverquant-0.0.4.tar.gz",
            "has_sig": false,
            "md5_digest": "b729591e2007ed2bbae12fd47605946a",
            "packagetype": "sdist",
            "python_version": "source",
            "requires_python": ">=3.7",
            "size": 26856,
            "upload_time": "2023-12-07T17:36:30",
            "upload_time_iso_8601": "2023-12-07T17:36:30.424576Z",
            "url": "https://files.pythonhosted.org/packages/a6/e2/5c78d4573c1a6d579a0582b4020a17f9b56fccc2a06af43eaa73f2977571/liverquant-0.0.4.tar.gz",
            "yanked": false,
            "yanked_reason": null
        }
    ],
    "upload_time": "2023-12-07 17:36:30",
    "github": true,
    "gitlab": false,
    "bitbucket": false,
    "codeberg": false,
    "github_user": "mfarzi",
    "github_project": "liverquant",
    "travis_ci": false,
    "coveralls": false,
    "github_actions": false,
    "lcname": "liverquant"
}
        
Elapsed time: 8.80540s