.. _custom_extractors:
Custom Feature Extractors
=========================
Slideflow includes several :ref:`pretrained feature extractors <mil>` for converting image tiles into feature vectors as well as tools to assist with building your own feature extractor. In this note, we'll walk through the process of building a custom feature extractor from both a PyTorch and Tensorflow model.
PyTorch
*******
Feature extractors are implemented as a subclass of :class:`slideflow.model.extractors._factory_torch.TorchFeatureExtractor`. The base class provides core functionality and helper methods for generating features from image tiles (dtype uint8) or whole-slide images (type :class:`slideflow.WSI`).
The initializer should create the feature extraction model and move it to the appropriate device (*i.e.* GPU). The model should be a :class:`torch.nn.Module` that accepts an image tensor as input and returns a feature tensor as output.
.. code-block:: python
# Import your custom torch.nn.Module,
# which generates features from an image.
from my_module import MyModel
from slideflow.model.extractors._factory_torch import TorchFeatureExtractor
class MyFeatureExtractor(TorchFeatureExtractor):
tag = 'my_feature_extractor' # Human-readable identifier
def __init__(self):
super().__init__()
# Create the device, move to GPU, and set in evaluation mode.
self.model = MyModel()
self.model.to('cuda')
self.model.eval()
Next, the initializer should set the number of features expected to be returned by the model.
.. code-block:: python
...
def __init__(self):
...
self.num_features = 1024
The initializer is also responsible for registering image preprocessing. The image preprocessing transformation, a function which converts a raw ``uint8`` image to a ``float32`` tensor for model input, should be stored in ``self.transform``. If the transformation standardizes the images, then the parameter ``self.preprocess_kwargs`` should be set to ``{'standardize': False}``, indicating that Slideflow should not perform any additional standardization. You can use the class method ``.build_transform()`` to use the standard preprocessing pipeline.
.. code-block:: python
from torchvision import transforms
...
def __init__(self):
...
# Image preprocessing.
self.transform = self.build_transform(img_size=256)
# Disable Slideflow standardization,
# as we are standardizing with transforms.Normalize
self.preprocess_kwargs = {'standardize': False}
The final required method is ``.dump_config()``, which returns a dictionary of configuration parameters needed to regenerate this class. It should return a dictionary with ``"class"`` and ``"kwargs"`` attributes. This configuration is saved to a JSON configuration file when generating bags for MIL training.
.. code-block:: python
...
def dump_config(self):
return self._dump_config(
class_name='my_module.MyFeatureExtractor'
)
The final class should look like this:
.. code-block:: python
from my_module import MyModel
from slideflow.model.extractors._factory_torch import TorchFeatureExtractor
from torchvision import transforms
class MyFeatureExtractor(TorchFeatureExtractor):
tag = 'my_feature_extractor' # Human-readable identifier
def __init__(self):
super().__init__()
# Create the device, move to GPU, and set in evaluation mode.
self.model = MyModel()
self.model.to('cuda')
self.model.eval()
self.num_features = 1024
# Image preprocessing.
self.transform = self.build_transform(img_size=256)
# Disable Slideflow standardization,
# as we are standardizing with transforms.Normalize
self.preprocess_kwargs = {'standardize': False}
def dump_config(self):
return self._dump_config(
class_name='my_module.MyFeatureExtractor'
)
You can then use the feature extractor for generating bags for MIL training, as described in :ref:`mil`.
.. code-block:: python
# Build the feature extractor.
myfeatures = MyFeatureExtractor()
# Load a dataset.
project = slideflow.load_project(...)
dataset = project.dataset(...)
# Generate bags.
project.generate_feature_bags(myfeatures, dataset)
You can also generate features across whole-slide images, returning a grid of features for each slide. The size of the returned grid reflects the slide's tile grid. For example, for a slide with 24 columns and 33 rows of tiles, the returned grid will have shape ``(24, 33, n_features)``.
.. code-block:: python
>>> myfeatures = MyFeatureExtractor()
>>> wsi = sf.WSI('path/to/wsi', tile_px=256, tile_um=302)
>>> features = myfeatures(wsi)
>>> features.shape
(24, 33, 1024)
Finally, the feature extractor can also be used to perform latent space analysis and generate mosaic maps, as described in :ref:`activations`.
Slideflow includes a registration system for keeping track of all available feature extractors. To register your feature extractor, use the :func:`slideflow.model.extractors.register_torch` decorator.
.. code-block:: python
from slideflow.model.extractors import register_torch
@register_torch
def my_feature_extractor(**kwargs):
return MyFeatureExtractor(**kwargs)
Once registered, a feature extractor can be built by name:
.. code-block:: python
import slideflow as sf
extractor = sf.build_feature_extractor('my_feature_extractor')
Tensorflow
**********
Tensorflow feature extractors are implemented very similarly to PyTorch feature extractors, extended from :class:`slideflow.model.extractors._tensorflow_base.TensorflowFeatureExtractor`.
The initializer should create the model and set the expected number of features.
.. code-block:: python
from my_module import MyModel
from slideflow.model.extractors._tensorflow_base import TensorflowFeatureExtractor
class MyFeatureExtractor(TensorflowFeatureExtractor):
tag = 'my_feature_extractor' # Unique identifier
def __init__(self):
super().__init__()
# Create the model.
self.model = MyModel()
self.num_features = 1024
.. |per_image_standardization| replace:: ``tf.image.per_image_standardization``
.. _per_image_standardization: https://www.tensorflow.org/api_docs/python/tf/image/per_image_standardization
The initializer is also responsible for registering image preprocessing and transformations. Preprocessing steps are stored in the ``.preprocess_kwargs`` dictionary, which should have the keys ``standardize`` and ``transform``. If ``standardize=True``, images will be standardized using |per_image_standardization|_. If ``transform`` is not None, it should be a callable that accepts a single image tensor and returns a transformed image tensor.
For example, to only perform standardization and no further preprocessing:
.. code-block:: python
...
def __init__(self):
...
# Image preprocessing.
self.preprocess_kwargs = {
'standardize': True,
'transform': None
}
To perform standardization and resize images to 256x256:
.. code-block:: python
import tensorflow as tf
@tf.function
def resize_256(x):
return = tf.image.resize(x, (resize_px, resize_px))
...
def __init__(self):
...
# Image preprocessing.
self.preprocess_kwargs = {
'standardize': True,
'transform': resize_256
}
The ``.dump_config()`` method should then be set, which is expected to return a dictionary of configuration parameters needed to regenerate this class. It should return a dictionary with ``"class"`` and ``"kwargs"`` attributes. This configuration is saved to a JSON configuration file when generating bags for MIL training.
.. code-block:: python
...
def dump_config(self):
return {
'class': 'MyFeatureExtractor',
'kwargs': {}
}
The final class should look like this:
.. code-block:: python
from my_module import MyModel
from slideflow.model.extractors._tensorflow_base import TensorflowFeatureExtractor
class MyFeatureExtractor(TensorflowFeatureExtractor):
tag = 'my_feature_extractor' # Unique identifier
def __init__(self):
super().__init__()
# Create the model.
self.model = MyModel()
self.num_features = 1024
# Image preprocessing.
self.preprocess_kwargs = {
'standardize': True,
'transform': None
}
def dump_config(self):
return {
'class': 'MyFeatureExtractor',
'kwargs': {}
}
As described above, this feature extractor can then be used to create bags for MIL training, generate features across whole-slide images, or perform feature space analysis across a dataset.
To register your feature extractor, use the :func:`slideflow.model.extractors.register_tensorflow` decorator.
.. code-block:: python
from slideflow.model.extractors import register_tf
@register_tf
def my_feature_extractor(**kwargs):
return MyFeatureExtractor(**kwargs)
...which will allow the feature extractor to be built by name:
.. code-block:: python
import slideflow as sf
extractor = sf.build_feature_extractor('my_feature_extractor')