From 630e9e55c4e9733326526c735f1e2617d9558ca9 Mon Sep 17 00:00:00 2001 From: pauladkisson Date: Thu, 28 Sep 2023 16:58:46 -0700 Subject: [PATCH 01/16] scanimagetiff_utils --- .../scanimagetiff_utils.py | 193 ++++++++++++++++++ 1 file changed, 193 insertions(+) create mode 100644 src/roiextractors/extractors/tiffimagingextractors/scanimagetiff_utils.py diff --git a/src/roiextractors/extractors/tiffimagingextractors/scanimagetiff_utils.py b/src/roiextractors/extractors/tiffimagingextractors/scanimagetiff_utils.py new file mode 100644 index 00000000..49f1c981 --- /dev/null +++ b/src/roiextractors/extractors/tiffimagingextractors/scanimagetiff_utils.py @@ -0,0 +1,193 @@ +import numpy as np + +from ...extraction_tools import PathType, get_package + + +def _get_scanimage_reader() -> type: + """Import the scanimage-tiff-reader package and return the ScanImageTiffReader class.""" + return get_package( + package_name="ScanImageTiffReader", installation_instructions="pip install scanimage-tiff-reader" + ).ScanImageTiffReader + + +def extract_extra_metadata( + file_path: PathType, +) -> dict: # TODO: Refactor neuroconv to reference this implementation to avoid duplication + """Extract metadata from a ScanImage TIFF file. + + Parameters + ---------- + file_path : PathType + Path to the TIFF file. + + Returns + ------- + extra_metadata: dict + Dictionary of metadata extracted from the TIFF file. + + Notes + ----- + Known to work on SI versions v3.8.0, v2019bR0, v2022.0.0, and v2023.0.0 + """ + ScanImageTiffReader = _get_scanimage_reader() + io = ScanImageTiffReader(str(file_path)) + extra_metadata = {} + for metadata_string in (io.description(iframe=0), io.metadata()): + metadata_dict = { + x.split("=")[0].strip(): x.split("=")[1].strip() + for x in metadata_string.replace("\n", "\r").split("\r") + if "=" in x + } + extra_metadata = dict(**extra_metadata, **metadata_dict) + return extra_metadata + + +def parse_matlab_vector(matlab_vector: str) -> list: + """Parse a MATLAB vector string into a list of integer values. + + Parameters + ---------- + matlab_vector : str + MATLAB vector string. + + Returns + ------- + vector: list of int + List of integer values. + + Raises + ------ + ValueError + If the MATLAB vector string cannot be parsed. + + Notes + ----- + MATLAB vector string is of the form "[1 2 3 ... N]" or "[1,2,3,...,N]" or "[1;2;3;...;N]". + There may or may not be whitespace between the values. Ex. "[1, 2, 3]" or "[1,2,3]". + """ + vector = matlab_vector.strip("[]") + if ";" in vector: + vector = vector.split(";") + elif "," in vector: + vector = vector.split(",") + elif " " in vector: + vector = vector.split(" ") + elif len(vector) == 1: + pass + else: + raise ValueError(f"Could not parse vector from {matlab_vector}.") + vector = [int(x.strip()) for x in vector if x != ""] + return vector + + +def parse_metadata(metadata: dict) -> dict: + """Parse metadata dictionary to extract relevant information and store it standard keys for ImagingExtractors. + + Currently supports + - sampling_frequency + - num_planes + - frames_per_slice + - channel_names + - num_channels + + Parameters + ---------- + metadata : dict + Dictionary of metadata extracted from the TIFF file. + + Returns + ------- + metadata_parsed: dict + Dictionary of parsed metadata. + + Notes + ----- + Known to work on SI versions v2019bR0, v2022.0.0, and v2023.0.0. Fails on v3.8.0. + SI.hChannels.channelsActive = string of MATLAB-style vector with channel integers (see parse_matlab_vector). + SI.hChannels.channelName = "{'channel_name_1' 'channel_name_2' ... 'channel_name_M'}" + where M is the number of channels (active or not). + """ + sampling_frequency = float(metadata["SI.hRoiManager.scanFrameRate"]) + num_planes = int(metadata["SI.hStackManager.numSlices"]) + frames_per_slice = int(metadata["SI.hStackManager.framesPerSlice"]) + active_channels = parse_matlab_vector(metadata["SI.hChannels.channelsActive"]) + channel_indices = np.array(active_channels) - 1 # Account for MATLAB indexing + channel_names = np.array(metadata["SI.hChannels.channelName"].split("'")[1::2]) + channel_names = channel_names[channel_indices].tolist() + num_channels = len(channel_names) + metadata_parsed = dict( + sampling_frequency=sampling_frequency, + num_channels=num_channels, + num_planes=num_planes, + frames_per_slice=frames_per_slice, + channel_names=channel_names, + ) + return metadata_parsed + + +def parse_metadata_v3_8(metadata: dict) -> dict: + """Parse metadata dictionary to extract relevant information and store it standard keys for ImagingExtractors. + + Requires old version of metadata (v3.8). + Currently supports + - sampling frequency + - num_channels + - num_planes + + Parameters + ---------- + metadata : dict + Dictionary of metadata extracted from the TIFF file. + + Returns + ------- + metadata_parsed: dict + Dictionary of parsed metadata. + """ + sampling_frequency = float(metadata["state.acq.frameRate"]) + num_channels = int(metadata["state.acq.numberOfChannelsSave"]) + num_planes = int(metadata["state.acq.numberOfZSlices"]) + metadata_parsed = dict( + sampling_frequency=sampling_frequency, + num_channels=num_channels, + num_planes=num_planes, + ) + return metadata_parsed + + +def extract_timestamps_from_file(file_path: PathType) -> np.ndarray: + """Extract the frame timestamps from a ScanImage TIFF file. + + Parameters + ---------- + file_path : PathType + Path to the TIFF file. + + Returns + ------- + timestamps : numpy.ndarray + Array of frame timestamps in seconds. + + Raises + ------ + AssertionError + If the frame timestamps are not found in the TIFF file. + + Notes + ----- + Known to work on SI versions v2019bR0, v2022.0.0, and v2023.0.0. Fails on v3.8.0. + """ + ScanImageTiffReader = _get_scanimage_reader() + io = ScanImageTiffReader(str(file_path)) + assert "frameTimestamps_sec" in io.description(iframe=0), "frameTimestamps_sec not found in TIFF file" + num_frames = io.shape()[0] + timestamps = np.zeros(num_frames) + for iframe in range(num_frames): + description = io.description(iframe=iframe) + description_lines = description.split("\n") + for line in description_lines: + if "frameTimestamps_sec" in line: + timestamps[iframe] = float(line.split("=")[1].strip()) + break + + return timestamps From ea3849de0cfab3870c251f319eba1fed4a077e5d Mon Sep 17 00:00:00 2001 From: pauladkisson Date: Thu, 28 Sep 2023 17:00:53 -0700 Subject: [PATCH 02/16] scanimagetiffimagingextractor --- .../scanimagetiffimagingextractor.py | 477 +++++++++++++++++- 1 file changed, 468 insertions(+), 9 deletions(-) diff --git a/src/roiextractors/extractors/tiffimagingextractors/scanimagetiffimagingextractor.py b/src/roiextractors/extractors/tiffimagingextractors/scanimagetiffimagingextractor.py index 83a51916..0ad12b43 100644 --- a/src/roiextractors/extractors/tiffimagingextractors/scanimagetiffimagingextractor.py +++ b/src/roiextractors/extractors/tiffimagingextractors/scanimagetiffimagingextractor.py @@ -6,29 +6,482 @@ Specialized extractor for reading TIFF files produced via ScanImage. """ from pathlib import Path -from typing import Optional, Tuple +from typing import Optional, Tuple, List, Iterable from warnings import warn - import numpy as np -from ...extraction_tools import PathType, FloatType, ArrayType, get_package +from ...extraction_tools import PathType, FloatType, ArrayType, DtypeType, get_package from ...imagingextractor import ImagingExtractor +from .scanimagetiff_utils import ( + extract_extra_metadata, + parse_metadata, + extract_timestamps_from_file, + _get_scanimage_reader, +) + + +class MultiPlaneImagingExtractor(ImagingExtractor): + """Class to combine multiple ImagingExtractor objects by depth plane.""" + + extractor_name = "MultiPlaneImaging" + installed = True + installatiuon_mesage = "" + + def __init__(self, imaging_extractors: List[ImagingExtractor]): + """Initialize a MultiPlaneImagingExtractor object from a list of ImagingExtractors. + + Parameters + ---------- + imaging_extractors: list of ImagingExtractor + list of imaging extractor objects + """ + super().__init__() + assert isinstance(imaging_extractors, list), "Enter a list of ImagingExtractor objects as argument" + assert all(isinstance(imaging_extractor, ImagingExtractor) for imaging_extractor in imaging_extractors) + self._check_consistency_between_imaging_extractors(imaging_extractors) + self._imaging_extractors = imaging_extractors + self._num_planes = len(imaging_extractors) + + # TODO: Add consistency check for channel_names when API is standardized + def _check_consistency_between_imaging_extractors(self, imaging_extractors: List[ImagingExtractor]): + """Check that essential properties are consistent between extractors so that they can be combined appropriately. + + Parameters + ---------- + imaging_extractors: list of ImagingExtractor + list of imaging extractor objects + + Raises + ------ + AssertionError + If any of the properties are not consistent between extractors. + + Notes + ----- + This method checks the following properties: + - sampling frequency + - image size + - number of channels + - channel names + - data type + - num_frames + """ + properties_to_check = dict( + get_sampling_frequency="The sampling frequency", + get_image_size="The size of a frame", + get_num_channels="The number of channels", + get_dtype="The data type", + get_num_frames="The number of frames", + ) + for method, property_message in properties_to_check.items(): + values = [getattr(extractor, method)() for extractor in imaging_extractors] + unique_values = set(tuple(v) if isinstance(v, Iterable) else v for v in values) + assert ( + len(unique_values) == 1 + ), f"{property_message} is not consistent over the files (found {unique_values})." + + def get_video(self, start_frame: Optional[int] = None, end_frame: Optional[int] = None) -> np.ndarray: + """Get the video frames. + + Parameters + ---------- + start_frame: int, optional + Start frame index (inclusive). + end_frame: int, optional + End frame index (exclusive). + + Returns + ------- + video: numpy.ndarray + The 3D video frames (num_rows, num_columns, num_planes). + """ + start_frame = start_frame if start_frame is not None else 0 + end_frame = end_frame if end_frame is not None else self.get_num_frames() + + video = np.zeros((end_frame - start_frame, *self.get_image_size()), self.get_dtype()) + for i, imaging_extractor in enumerate(self._imaging_extractors): + video[..., i] = imaging_extractor.get_video(start_frame, end_frame) + return video + + def get_frames(self, frame_idxs: ArrayType) -> np.ndarray: + """Get specific video frames from indices (not necessarily continuous). + + Parameters + ---------- + frame_idxs: array-like + Indices of frames to return. + + Returns + ------- + frames: numpy.ndarray + The 3D video frames (num_rows, num_columns, num_planes). + """ + if isinstance(frame_idxs, int): + frame_idxs = [frame_idxs] + if not all(np.diff(frame_idxs) == 1): + frames = np.zeros((len(frame_idxs), *self.get_image_size()), self.get_dtype()) + for i, imaging_extractor in enumerate(self._imaging_extractors): + frames[..., i] = imaging_extractor.get_frames(frame_idxs) + else: + return self.get_video(start_frame=frame_idxs[0], end_frame=frame_idxs[-1] + 1) + + def get_image_size(self) -> Tuple: + """Get the size of a single frame. + + Returns + ------- + image_size: tuple + The size of a single frame (num_rows, num_columns, num_planes). + """ + image_size = (*self._imaging_extractors[0].get_image_size(), self.get_num_planes()) + return image_size + + def get_num_planes(self) -> int: + """Get the number of depth planes. -def _get_scanimage_reader() -> type: - """Import the scanimage-tiff-reader package and return the ScanImageTiffReader class.""" - return get_package( - package_name="ScanImageTiffReader", installation_instructions="pip install scanimage-tiff-reader" - ).ScanImageTiffReader + Returns + ------- + _num_planes: int + The number of depth planes. + """ + return self._num_planes + def get_num_frames(self) -> int: + return self._imaging_extractors[0].get_num_frames() -class ScanImageTiffImagingExtractor(ImagingExtractor): + def get_sampling_frequency(self) -> float: + return self._imaging_extractors[0].get_sampling_frequency() + + def get_channel_names(self) -> list: + return self._imaging_extractors[0].get_channel_names() + + def get_num_channels(self) -> int: + return self._imaging_extractors[0].get_num_channels() + + def get_dtype(self) -> DtypeType: + return self._imaging_extractors[0].get_dtype() + + +class ScanImageTiffMultiPlaneImagingExtractor(MultiPlaneImagingExtractor): + """Specialized extractor for reading multi-plane (volumetric) TIFF files produced via ScanImage.""" + + extractor_name = "ScanImageTiffMultiPlaneImaging" + is_writable = True + mode = "file" + + def __init__( + self, + file_path: PathType, + channel_name: Optional[str] = None, + ) -> None: + self.file_path = Path(file_path) + self.metadata = extract_extra_metadata(file_path) + parsed_metadata = parse_metadata(self.metadata) + num_planes = parsed_metadata["num_planes"] + channel_names = parsed_metadata["channel_names"] + if channel_name is None: + channel_name = channel_names[0] + imaging_extractors = [] + for plane in range(num_planes): + imaging_extractor = ScanImageTiffSinglePlaneImagingExtractor( + file_path=file_path, channel_name=channel_name, plane_name=str(plane) + ) + imaging_extractors.append(imaging_extractor) + super().__init__(imaging_extractors=imaging_extractors) + assert all( + imaging_extractor.get_num_planes() == self._num_planes for imaging_extractor in imaging_extractors + ), "All imaging extractors must have the same number of planes." + + +class ScanImageTiffSinglePlaneImagingExtractor(ImagingExtractor): """Specialized extractor for reading TIFF files produced via ScanImage.""" extractor_name = "ScanImageTiffImaging" is_writable = True mode = "file" + @classmethod + def get_channel_names(cls, file_path): + """Get the channel names from a TIFF file produced by ScanImage. + + Parameters + ---------- + file_path : PathType + Path to the TIFF file. + + Returns + ------- + channel_names: list + List of channel names. + """ + metadata = extract_extra_metadata(file_path) + parsed_metadata = parse_metadata(metadata) + channel_names = parsed_metadata["channel_names"] + return channel_names + + @classmethod + def get_plane_names(cls, file_path): + """Get the plane names from a TIFF file produced by ScanImage. + + Parameters + ---------- + file_path : PathType + Path to the TIFF file. + + Returns + ------- + plane_names: list + List of plane names. + """ + metadata = extract_extra_metadata(file_path) + parsed_metadata = parse_metadata(metadata) + num_planes = parsed_metadata["num_planes"] + plane_names = [f"{i}" for i in range(num_planes)] + return plane_names + + def __init__( + self, + file_path: PathType, + channel_name: str, + plane_name: str, + ) -> None: + """Create a ScanImageTiffImagingExtractor instance from a TIFF file produced by ScanImage. + + The underlying data is stored in a round-robin format collapsed into 3 dimensions (frames, rows, columns). + I.e. the first frame of each channel and each plane is stored, and then the second frame of each channel and + each plane, etc. + If framesPerSlice > 1, then multiple frames are acquired per slice before moving to the next slice. + Ex. for 2 channels, 2 planes, and 2 framesPerSlice: + ``` + [channel_1_plane_1_frame_1, channel_2_plane_1_frame_1, channel_1_plane_1_frame_2, channel_2_plane_1_frame_2, + channel_1_plane_2_frame_1, channel_2_plane_2_frame_1, channel_1_plane_2_frame_2, channel_2_plane_2_frame_2, + channel_1_plane_1_frame_3, channel_2_plane_1_frame_3, channel_1_plane_1_frame_4, channel_2_plane_1_frame_4, + channel_1_plane_2_frame_3, channel_2_plane_2_frame_3, channel_1_plane_2_frame_4, channel_2_plane_2_frame_4, ... + channel_1_plane_1_frame_N, channel_2_plane_1_frame_N, channel_1_plane_2_frame_N, channel_2_plane_2_frame_N] + ``` + This file structured is accessed by ScanImageTiffImagingExtractor for a single channel and plane. + + Parameters + ---------- + file_path : PathType + Path to the TIFF file. + channel_name : str + Name of the channel for this extractor (default=None). + plane_name : str + Name of the plane for this extractor (default=None). + """ + self.file_path = Path(file_path) + self.metadata = extract_extra_metadata(file_path) + parsed_metadata = parse_metadata(self.metadata) + self._sampling_frequency = parsed_metadata["sampling_frequency"] + self._num_channels = parsed_metadata["num_channels"] + self._num_planes = parsed_metadata["num_planes"] + self._frames_per_slice = parsed_metadata["frames_per_slice"] + self._channel_names = parsed_metadata["channel_names"] + self._plane_names = [f"{i}" for i in range(self._num_planes)] + self.channel_name = channel_name + self.plane_name = plane_name + if channel_name not in self._channel_names: + raise ValueError(f"Channel name ({channel_name}) not found in channel names ({self._channel_names}).") + self.channel = self._channel_names.index(channel_name) + if plane_name not in self._plane_names: + raise ValueError(f"Plane name ({plane_name}) not found in plane names ({self._plane_names}).") + self.plane = self._plane_names.index(plane_name) + + valid_suffixes = [".tiff", ".tif", ".TIFF", ".TIF"] + if self.file_path.suffix not in valid_suffixes: + suffix_string = ", ".join(valid_suffixes[:-1]) + f", or {valid_suffixes[-1]}" + warn( + f"Suffix ({self.file_path.suffix}) is not of type {suffix_string}! " + f"The {self.extractor_name}Extractor may not be appropriate for the file." + ) + ScanImageTiffReader = _get_scanimage_reader() + with ScanImageTiffReader(str(self.file_path)) as io: + shape = io.shape() # [frames, rows, columns] + if len(shape) == 3: + self._total_num_frames, self._num_rows, self._num_columns = shape + self._num_raw_per_plane = self._frames_per_slice * self._num_channels + self._num_raw_per_cycle = self._num_raw_per_plane * self._num_planes + self._num_frames = self._total_num_frames // (self._num_planes * self._num_channels) + self._num_cycles = self._total_num_frames // self._num_raw_per_cycle + else: + raise NotImplementedError( + "Extractor cannot handle 4D ScanImageTiff data. Please raise an issue to request this feature: " + "https://github.com/catalystneuro/roiextractors/issues " + ) + timestamps = extract_timestamps_from_file(file_path) + index = [self.frame_to_raw_index(iframe) for iframe in range(self._num_frames)] + self._times = timestamps[index] + + def get_frames(self, frame_idxs: ArrayType) -> np.ndarray: + """Get specific video frames from indices (not necessarily continuous). + + Parameters + ---------- + frame_idxs: array-like + Indices of frames to return. + + Returns + ------- + frames: numpy.ndarray + The video frames. + """ + if isinstance(frame_idxs, int): + frame_idxs = [frame_idxs] + self.check_frame_inputs(frame_idxs[-1]) + + if not all(np.diff(frame_idxs) == 1): + return np.concatenate([self._get_single_frame(frame=idx) for idx in frame_idxs]) + else: + return self.get_video(start_frame=frame_idxs[0], end_frame=frame_idxs[-1] + 1) + + # Data accessed through an open ScanImageTiffReader io gets scrambled if there are multiple calls. + # Thus, open fresh io in context each time something is needed. + def _get_single_frame(self, frame: int) -> np.ndarray: + """Get a single frame of data from the TIFF file. + + Parameters + ---------- + frame : int + The index of the frame to retrieve. + + Returns + ------- + frame: numpy.ndarray + The frame of data. + """ + self.check_frame_inputs(frame) + ScanImageTiffReader = _get_scanimage_reader() + raw_index = self.frame_to_raw_index(frame) + with ScanImageTiffReader(str(self.file_path)) as io: + return io.data(beg=raw_index, end=raw_index + 1) + + def get_video(self, start_frame=None, end_frame=None) -> np.ndarray: + """Get the video frames. + + Parameters + ---------- + start_frame: int, optional + Start frame index (inclusive). + end_frame: int, optional + End frame index (exclusive). + + Returns + ------- + video: numpy.ndarray + The video frames. + """ + if start_frame is None: + start_frame = 0 + if end_frame is None: + end_frame = self._num_frames + end_frame_inclusive = end_frame - 1 + self.check_frame_inputs(end_frame_inclusive) + self.check_frame_inputs(start_frame) + raw_start = self.frame_to_raw_index(start_frame) + raw_end_inclusive = self.frame_to_raw_index(end_frame_inclusive) # frame_to_raw_index requires inclusive frame + raw_end = raw_end_inclusive + 1 + + ScanImageTiffReader = _get_scanimage_reader() + with ScanImageTiffReader(filename=str(self.file_path)) as io: + raw_video = io.data(beg=raw_start, end=raw_end) + + start_cycle = np.ceil(start_frame / self._frames_per_slice).astype("int") + end_cycle = end_frame // self._frames_per_slice + num_cycles = end_cycle - start_cycle + start_frame_in_cycle = start_frame % self._frames_per_slice + end_frame_in_cycle = end_frame % self._frames_per_slice + start_left_in_cycle = (self._frames_per_slice - start_frame_in_cycle) % self._frames_per_slice + end_left_in_cycle = (self._frames_per_slice - end_frame_in_cycle) % self._frames_per_slice + index = [] + for j in range(start_left_in_cycle): # Add remaining frames from first (incomplete) cycle + index.append(j * self._num_channels) + for i in range(num_cycles): + for j in range(self._frames_per_slice): + index.append( + (j - start_frame_in_cycle) * self._num_channels + + (i + bool(start_left_in_cycle)) * self._num_raw_per_cycle + ) + for j in range(end_left_in_cycle): # Add remaining frames from last (incomplete) cycle) + index.append((j - start_frame_in_cycle) * self._num_channels + num_cycles * self._num_raw_per_cycle) + video = raw_video[index] + return video + + def get_image_size(self) -> Tuple[int, int]: + return (self._num_rows, self._num_columns) + + def get_num_frames(self) -> int: + return self._num_frames + + def get_sampling_frequency(self) -> float: + return self._sampling_frequency + + def get_num_channels(self) -> int: + return self._num_channels + + def get_num_planes(self) -> int: + return self._num_planes + + def get_dtype(self) -> DtypeType: + return self.get_frames(0).dtype + + def check_frame_inputs(self, frame) -> None: + if frame >= self._num_frames: + raise ValueError(f"Frame index ({frame}) exceeds number of frames ({self._num_frames}).") + if frame < 0: + raise ValueError(f"Frame index ({frame}) must be greater than or equal to 0.") + + def frame_to_raw_index(self, frame): + """Convert a frame index to the raw index in the TIFF file. + + Parameters + ---------- + frame : int + The index of the frame to retrieve. + + Returns + ------- + raw_index: int + The raw index of the frame in the TIFF file. + + Notes + ----- + The underlying data is stored in a round-robin format collapsed into 3 dimensions (frames, rows, columns). + I.e. the first frame of each channel and each plane is stored, and then the second frame of each channel and + each plane, etc. + If framesPerSlice > 1, then multiple frames are acquired per slice before moving to the next slice. + Ex. for 2 channels, 2 planes, and 2 framesPerSlice: + ``` + [channel_1_plane_1_frame_1, channel_2_plane_1_frame_1, channel_1_plane_1_frame_2, channel_2_plane_1_frame_2, + channel_1_plane_2_frame_1, channel_2_plane_2_frame_1, channel_1_plane_2_frame_2, channel_2_plane_2_frame_2, + channel_1_plane_1_frame_3, channel_2_plane_1_frame_3, channel_1_plane_1_frame_4, channel_2_plane_1_frame_4, + channel_1_plane_2_frame_3, channel_2_plane_2_frame_3, channel_1_plane_2_frame_4, channel_2_plane_2_frame_4, ... + channel_1_plane_1_frame_N, channel_2_plane_1_frame_N, channel_1_plane_2_frame_N, channel_2_plane_2_frame_N] + ``` + """ + cycle = frame // self._frames_per_slice + frame_in_cycle = frame % self._frames_per_slice + raw_index = ( + cycle * self._num_raw_per_cycle + + self.plane * self._num_raw_per_plane + + frame_in_cycle * self._num_channels + + self.channel + ) + return raw_index + + +class ScanImageTiffImagingExtractor(ImagingExtractor): # TODO: Remove this extractor on/after December 2023 + """Specialized extractor for reading TIFF files produced via ScanImage. + + This implementation is for legacy purposes and is not recommended for use. + Please use ScanImageTiffSinglePlaneImagingExtractor or ScanImageTiffMultiPlaneImagingExtractor instead. + """ + + extractor_name = "ScanImageTiffImaging" + is_writable = True + mode = "file" + def __init__( self, file_path: PathType, @@ -47,6 +500,12 @@ def __init__( sampling_frequency : float The frequency at which the frames were sampled, in Hz. """ + deprecation_message = """ + This extractor is being deprecated on or after December 2023 in favor of + ScanImageTiffMultiPlaneImagingExtractor or ScanImageTiffSinglePlaneImagingExtractor. Please use one of these + extractors instead. + """ + warn(deprecation_message, category=FutureWarning) ScanImageTiffReader = _get_scanimage_reader() super().__init__() From a3d82621e9dcecc9d01bcb3f663bc0e0d6b3bdb1 Mon Sep 17 00:00:00 2001 From: pauladkisson Date: Thu, 28 Sep 2023 17:02:12 -0700 Subject: [PATCH 03/16] init --- .../extractors/tiffimagingextractors/__init__.py | 12 ++++++++++-- 1 file changed, 10 insertions(+), 2 deletions(-) diff --git a/src/roiextractors/extractors/tiffimagingextractors/__init__.py b/src/roiextractors/extractors/tiffimagingextractors/__init__.py index b8f5cf54..3f3a3618 100644 --- a/src/roiextractors/extractors/tiffimagingextractors/__init__.py +++ b/src/roiextractors/extractors/tiffimagingextractors/__init__.py @@ -16,7 +16,11 @@ TiffImagingExtractor A ImagingExtractor for TIFF files. ScanImageTiffImagingExtractor - Specialized extractor for reading TIFF files produced via ScanImage. + Legacy extractor for reading TIFF files produced via ScanImage v3.8. +ScanImageTiffSinglePlaneImagingExtractor + Specialized extractor for reading single-plane TIFF files produced via ScanImage. +ScanImageTiffMultiPlaneImagingExtractor + Specialized extractor for reading multi-plane TIFF files produced via ScanImage. BrukerTiffMultiPlaneImagingExtractor Specialized extractor for reading TIFF files produced via Bruker. BrukerTiffSinglePlaneImagingExtractor @@ -25,6 +29,10 @@ Specialized extractor for reading TIFF files produced via Micro-Manager. """ from .tiffimagingextractor import TiffImagingExtractor -from .scanimagetiffimagingextractor import ScanImageTiffImagingExtractor +from .scanimagetiffimagingextractor import ( + ScanImageTiffImagingExtractor, + ScanImageTiffMultiPlaneImagingExtractor, + ScanImageTiffSinglePlaneImagingExtractor, +) from .brukertiffimagingextractor import BrukerTiffMultiPlaneImagingExtractor, BrukerTiffSinglePlaneImagingExtractor from .micromanagertiffimagingextractor import MicroManagerTiffImagingExtractor From ead93e4d670b0197da82b20365297c3cf5b2f2b7 Mon Sep 17 00:00:00 2001 From: pauladkisson Date: Thu, 28 Sep 2023 17:02:57 -0700 Subject: [PATCH 04/16] extractorlist --- src/roiextractors/extractorlist.py | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/src/roiextractors/extractorlist.py b/src/roiextractors/extractorlist.py index 7241de80..dff3136a 100644 --- a/src/roiextractors/extractorlist.py +++ b/src/roiextractors/extractorlist.py @@ -15,6 +15,8 @@ from .extractors.tiffimagingextractors import ( TiffImagingExtractor, ScanImageTiffImagingExtractor, + ScanImageTiffSinglePlaneImagingExtractor, + ScanImageTiffMultiPlaneImagingExtractor, BrukerTiffMultiPlaneImagingExtractor, BrukerTiffSinglePlaneImagingExtractor, MicroManagerTiffImagingExtractor, @@ -25,13 +27,14 @@ from .extractors.miniscopeimagingextractor import MiniscopeImagingExtractor from .multisegmentationextractor import MultiSegmentationExtractor from .multiimagingextractor import MultiImagingExtractor -from .volumetricimagingextractor import VolumetricImagingExtractor imaging_extractor_full_list = [ NumpyImagingExtractor, Hdf5ImagingExtractor, TiffImagingExtractor, ScanImageTiffImagingExtractor, + ScanImageTiffSinglePlaneImagingExtractor, + ScanImageTiffMultiPlaneImagingExtractor, BrukerTiffMultiPlaneImagingExtractor, BrukerTiffSinglePlaneImagingExtractor, MicroManagerTiffImagingExtractor, @@ -40,7 +43,6 @@ SbxImagingExtractor, NumpyMemmapImagingExtractor, MemmapImagingExtractor, - VolumetricImagingExtractor, ] segmentation_extractor_full_list = [ From bfb9ceeb4771a95afa3d7fb7b766a513de00056c Mon Sep 17 00:00:00 2001 From: pauladkisson Date: Thu, 28 Sep 2023 17:04:26 -0700 Subject: [PATCH 05/16] test_scanimage_utils --- tests/test_scanimage_utils.py | 0 1 file changed, 0 insertions(+), 0 deletions(-) create mode 100644 tests/test_scanimage_utils.py diff --git a/tests/test_scanimage_utils.py b/tests/test_scanimage_utils.py new file mode 100644 index 00000000..e69de29b From ae97296ab273c075c89dcd2278c3528b77b9eb1c Mon Sep 17 00:00:00 2001 From: pauladkisson Date: Thu, 28 Sep 2023 17:05:33 -0700 Subject: [PATCH 06/16] test_scanimagetiffimagingextractor --- tests/test_scanimagetiffimagingextractor.py | 225 ++++++++++++++++++++ 1 file changed, 225 insertions(+) create mode 100644 tests/test_scanimagetiffimagingextractor.py diff --git a/tests/test_scanimagetiffimagingextractor.py b/tests/test_scanimagetiffimagingextractor.py new file mode 100644 index 00000000..845fc443 --- /dev/null +++ b/tests/test_scanimagetiffimagingextractor.py @@ -0,0 +1,225 @@ +import pytest +from numpy.testing import assert_array_equal +from ScanImageTiffReader import ScanImageTiffReader +from roiextractors import ScanImageTiffSinglePlaneImagingExtractor, ScanImageTiffMultiPlaneImagingExtractor + +from .setup_paths import OPHYS_DATA_PATH + + +@pytest.fixture(scope="module") +def file_path(): + return OPHYS_DATA_PATH / "imaging_datasets" / "ScanImage" / "scanimage_20220923_roi.tif" + + +@pytest.fixture(scope="module") +def expected_properties(): + return dict( + sampling_frequency=29.1248, + num_channels=2, + num_planes=2, + frames_per_slice=2, + channel_names=["Channel 1", "Channel 4"], + image_size=(528, 256), + num_frames=6, + dtype="int16", + ) + + +@pytest.fixture( + scope="module", + params=[ + dict(channel_name="Channel 1", plane_name="0"), + dict(channel_name="Channel 1", plane_name="1"), + dict(channel_name="Channel 4", plane_name="0"), + dict(channel_name="Channel 4", plane_name="1"), + ], +) +def scan_image_tiff_single_plane_imaging_extractor(request, file_path): + return ScanImageTiffSinglePlaneImagingExtractor(file_path=file_path, **request.param) + + +@pytest.mark.parametrize("channel_name, plane_name", [("Invalid Channel", "0"), ("Channel 1", "Invalid Plane")]) +def test_ScanImageTiffSinglePlaneImagingExtractor__init__invalid(file_path, channel_name, plane_name): + with pytest.raises(ValueError): + ScanImageTiffSinglePlaneImagingExtractor(file_path=file_path, channel_name=channel_name, plane_name=plane_name) + + +@pytest.mark.parametrize("frame_idxs", (0, [0, 1, 2], [0, 2, 5])) +def test_get_frames(scan_image_tiff_single_plane_imaging_extractor, frame_idxs, expected_properties): + frames = scan_image_tiff_single_plane_imaging_extractor.get_frames(frame_idxs=frame_idxs) + file_path = str(scan_image_tiff_single_plane_imaging_extractor.file_path) + plane = scan_image_tiff_single_plane_imaging_extractor.plane + channel = scan_image_tiff_single_plane_imaging_extractor.channel + num_planes = expected_properties["num_planes"] + num_channels = expected_properties["num_channels"] + frames_per_slice = expected_properties["frames_per_slice"] + if isinstance(frame_idxs, int): + frame_idxs = [frame_idxs] + raw_idxs = [] + for idx in frame_idxs: + cycle = idx // frames_per_slice + frame_in_cycle = idx % frames_per_slice + raw_idx = ( + cycle * num_planes * num_channels * frames_per_slice + + plane * num_channels * frames_per_slice + + num_channels * frame_in_cycle + + channel + ) + raw_idxs.append(raw_idx) + with ScanImageTiffReader(file_path) as io: + assert_array_equal(frames, io.data()[raw_idxs]) + + +@pytest.mark.parametrize("frame_idxs", ([-1], [50])) +def test_get_frames_invalid(scan_image_tiff_single_plane_imaging_extractor, frame_idxs): + with pytest.raises(ValueError): + scan_image_tiff_single_plane_imaging_extractor.get_frames(frame_idxs=frame_idxs) + + +@pytest.mark.parametrize("frame_idx", (1, 3, 5)) +def test_get_single_frame(scan_image_tiff_single_plane_imaging_extractor, expected_properties, frame_idx): + frame = scan_image_tiff_single_plane_imaging_extractor._get_single_frame(frame=frame_idx) + file_path = str(scan_image_tiff_single_plane_imaging_extractor.file_path) + plane = scan_image_tiff_single_plane_imaging_extractor.plane + channel = scan_image_tiff_single_plane_imaging_extractor.channel + num_planes = expected_properties["num_planes"] + num_channels = expected_properties["num_channels"] + frames_per_slice = expected_properties["frames_per_slice"] + cycle = frame_idx // frames_per_slice + frame_in_cycle = frame_idx % frames_per_slice + raw_idx = ( + cycle * num_planes * num_channels * frames_per_slice + + plane * num_channels * frames_per_slice + + num_channels * frame_in_cycle + + channel + ) + with ScanImageTiffReader(file_path) as io: + assert_array_equal(frame, io.data()[raw_idx : raw_idx + 1]) + + +@pytest.mark.parametrize("frame", (-1, 50)) +def test_get_single_frame_invalid(scan_image_tiff_single_plane_imaging_extractor, frame): + with pytest.raises(ValueError): + scan_image_tiff_single_plane_imaging_extractor._get_single_frame(frame=frame) + + +@pytest.mark.parametrize("start_frame, end_frame", [(0, None), (None, 6), (1, 4), (0, 6)]) +def test_get_video( + scan_image_tiff_single_plane_imaging_extractor, + expected_properties, + start_frame, + end_frame, +): + video = scan_image_tiff_single_plane_imaging_extractor.get_video(start_frame=start_frame, end_frame=end_frame) + if start_frame is None: + start_frame = 0 + if end_frame is None: + end_frame = expected_properties["num_frames"] + file_path = str(scan_image_tiff_single_plane_imaging_extractor.file_path) + plane = scan_image_tiff_single_plane_imaging_extractor.plane + channel = scan_image_tiff_single_plane_imaging_extractor.channel + num_planes = expected_properties["num_planes"] + num_channels = expected_properties["num_channels"] + frames_per_slice = expected_properties["frames_per_slice"] + raw_idxs = [] + for idx in range(start_frame, end_frame): + cycle = idx // frames_per_slice + frame_in_cycle = idx % frames_per_slice + raw_idx = ( + cycle * num_planes * num_channels * frames_per_slice + + plane * num_channels * frames_per_slice + + num_channels * frame_in_cycle + + channel + ) + raw_idxs.append(raw_idx) + with ScanImageTiffReader(file_path) as io: + assert_array_equal(video, io.data()[raw_idxs]) + + +@pytest.mark.parametrize("start_frame, end_frame", [(-1, 2), (0, 50)]) +def test_get_video_invalid( + scan_image_tiff_single_plane_imaging_extractor, + start_frame, + end_frame, +): + with pytest.raises(ValueError): + scan_image_tiff_single_plane_imaging_extractor.get_video(start_frame=start_frame, end_frame=end_frame) + + +def test_get_image_size(scan_image_tiff_single_plane_imaging_extractor, expected_properties): + image_size = scan_image_tiff_single_plane_imaging_extractor.get_image_size() + assert image_size == expected_properties["image_size"] + + +def test_get_num_frames(scan_image_tiff_single_plane_imaging_extractor, expected_properties): + num_frames = scan_image_tiff_single_plane_imaging_extractor.get_num_frames() + assert num_frames == expected_properties["num_frames"] + + +def test_get_sampling_frequency(scan_image_tiff_single_plane_imaging_extractor, expected_properties): + sampling_frequency = scan_image_tiff_single_plane_imaging_extractor.get_sampling_frequency() + assert sampling_frequency == expected_properties["sampling_frequency"] + + +def test_get_num_channels(scan_image_tiff_single_plane_imaging_extractor, expected_properties): + num_channels = scan_image_tiff_single_plane_imaging_extractor.get_num_channels() + assert num_channels == expected_properties["num_channels"] + + +def test_get_channel_names(scan_image_tiff_single_plane_imaging_extractor, expected_properties): + file_path = str(scan_image_tiff_single_plane_imaging_extractor.file_path) + channel_names = ScanImageTiffSinglePlaneImagingExtractor.get_channel_names(file_path) + assert channel_names == expected_properties["channel_names"] + + +def test_get_num_planes(scan_image_tiff_single_plane_imaging_extractor, expected_properties): + num_planes = scan_image_tiff_single_plane_imaging_extractor.get_num_planes() + assert num_planes == expected_properties["num_planes"] + + +def test_get_dtype(scan_image_tiff_single_plane_imaging_extractor, expected_properties): + dtype = scan_image_tiff_single_plane_imaging_extractor.get_dtype() + assert dtype == expected_properties["dtype"] + + +def test_check_frame_inputs_valid(scan_image_tiff_single_plane_imaging_extractor): + scan_image_tiff_single_plane_imaging_extractor.check_frame_inputs(frame=0) + + +def test_check_frame_inputs_invalid(scan_image_tiff_single_plane_imaging_extractor, expected_properties): + num_frames = expected_properties["num_frames"] + with pytest.raises(ValueError): + scan_image_tiff_single_plane_imaging_extractor.check_frame_inputs(frame=num_frames + 1) + + +@pytest.mark.parametrize("frame", (0, 3, 5)) +def test_frame_to_raw_index( + scan_image_tiff_single_plane_imaging_extractor, + frame, + expected_properties, +): + raw_index = scan_image_tiff_single_plane_imaging_extractor.frame_to_raw_index(frame=frame) + plane = scan_image_tiff_single_plane_imaging_extractor.plane + channel = scan_image_tiff_single_plane_imaging_extractor.channel + num_planes = expected_properties["num_planes"] + num_channels = expected_properties["num_channels"] + frames_per_slice = expected_properties["frames_per_slice"] + cycle = frame // frames_per_slice + frame_in_cycle = frame % frames_per_slice + expected_index = ( + cycle * num_planes * num_channels * frames_per_slice + + plane * num_channels * frames_per_slice + + num_channels * frame_in_cycle + + channel + ) + assert raw_index == expected_index + + +def test_ScanImageTiffMultiPlaneImagingExtractor__init__(file_path): + extractor = ScanImageTiffMultiPlaneImagingExtractor(file_path=file_path) + assert extractor.file_path == file_path + + +def test_ScanImageTiffMultiPlaneImagingExtractor__init__invalid(file_path): + with pytest.raises(ValueError): + ScanImageTiffMultiPlaneImagingExtractor(file_path=file_path, channel_name="Invalid Channel") From 00d5b4c2abcf4223e7da3a9ef75eaf5c3734f295 Mon Sep 17 00:00:00 2001 From: pauladkisson Date: Thu, 28 Sep 2023 17:07:38 -0700 Subject: [PATCH 07/16] extractorlist (fixed bug) --- src/roiextractors/extractorlist.py | 1 + 1 file changed, 1 insertion(+) diff --git a/src/roiextractors/extractorlist.py b/src/roiextractors/extractorlist.py index dff3136a..27f5f050 100644 --- a/src/roiextractors/extractorlist.py +++ b/src/roiextractors/extractorlist.py @@ -27,6 +27,7 @@ from .extractors.miniscopeimagingextractor import MiniscopeImagingExtractor from .multisegmentationextractor import MultiSegmentationExtractor from .multiimagingextractor import MultiImagingExtractor +from .volumetricimagingextractor import VolumetricImagingExtractor imaging_extractor_full_list = [ NumpyImagingExtractor, From 851fea4c54b3d67ed8044ad88d6f9c9e3bdec97d Mon Sep 17 00:00:00 2001 From: pauladkisson Date: Thu, 28 Sep 2023 17:10:31 -0700 Subject: [PATCH 08/16] extractorlist (fixed bug) --- src/roiextractors/extractorlist.py | 1 + 1 file changed, 1 insertion(+) diff --git a/src/roiextractors/extractorlist.py b/src/roiextractors/extractorlist.py index 27f5f050..73d0bf6e 100644 --- a/src/roiextractors/extractorlist.py +++ b/src/roiextractors/extractorlist.py @@ -44,6 +44,7 @@ SbxImagingExtractor, NumpyMemmapImagingExtractor, MemmapImagingExtractor, + VolumetricImagingExtractor, ] segmentation_extractor_full_list = [ From 53acb1d7680afbbc64a11e08d7af08ccf36e3774 Mon Sep 17 00:00:00 2001 From: pauladkisson Date: Thu, 28 Sep 2023 17:17:16 -0700 Subject: [PATCH 09/16] test_scanimage_utils --- tests/test_scanimage_utils.py | 128 ++++++++++++++++++++++++++++++++++ 1 file changed, 128 insertions(+) diff --git a/tests/test_scanimage_utils.py b/tests/test_scanimage_utils.py index e69de29b..4b65b0cf 100644 --- a/tests/test_scanimage_utils.py +++ b/tests/test_scanimage_utils.py @@ -0,0 +1,128 @@ +import pytest +from numpy.testing import assert_array_equal +from ScanImageTiffReader import ScanImageTiffReader +from roiextractors.extractors.tiffimagingextractors.scanimagetiff_utils import ( + _get_scanimage_reader, + extract_extra_metadata, + parse_matlab_vector, + parse_metadata, + parse_metadata_v3_8, + extract_timestamps_from_file, +) + +from .setup_paths import OPHYS_DATA_PATH + + +def test_get_scanimage_reader(): + ScanImageTiffReader = _get_scanimage_reader() + assert ScanImageTiffReader is not None + + +@pytest.mark.parametrize( + "filename, expected_key, expected_value", + [ + ("sample_scanimage_version_3_8.tiff", "state.software.version", "3.8"), + ("scanimage_20220801_single.tif", "SI.VERSION_MAJOR", "2022"), + ("scanimage_20220923_roi.tif", "SI.VERSION_MAJOR", "2023"), + ], +) +def test_extract_extra_metadata(filename, expected_key, expected_value): + file_path = OPHYS_DATA_PATH / "imaging_datasets" / "ScanImage" / filename + metadata = extract_extra_metadata(file_path) + assert metadata[expected_key] == expected_value + + +@pytest.mark.parametrize( + "matlab_vector, expected_vector", + [ + ("[1 2 3]", [1, 2, 3]), + ("[1,2,3]", [1, 2, 3]), + ("[1, 2, 3]", [1, 2, 3]), + ("[1;2;3]", [1, 2, 3]), + ("[1; 2; 3]", [1, 2, 3]), + ], +) +def test_parse_matlab_vector(matlab_vector, expected_vector): + vector = parse_matlab_vector(matlab_vector) + assert vector == expected_vector + + +@pytest.mark.parametrize( + "filename, expected_metadata", + [ + ( + "scanimage_20220801_single.tif", + { + "sampling_frequency": 15.2379, + "num_channels": 1, + "num_planes": 20, + "frames_per_slice": 24, + "channel_names": ["Channel 1"], + }, + ), + ( + "scanimage_20220923_roi.tif", + { + "sampling_frequency": 29.1248, + "num_channels": 2, + "num_planes": 2, + "frames_per_slice": 2, + "channel_names": ["Channel 1", "Channel 4"], + }, + ), + ], +) +def test_parse_metadata(filename, expected_metadata): + file_path = OPHYS_DATA_PATH / "imaging_datasets" / "ScanImage" / filename + metadata = extract_extra_metadata(file_path) + metadata = parse_metadata(metadata) + assert metadata == expected_metadata + + +def test_parse_metadata_v3_8(): + file_path = OPHYS_DATA_PATH / "imaging_datasets" / "ScanImage" / "sample_scanimage_version_3_8.tiff" + metadata = extract_extra_metadata(file_path) + metadata = parse_metadata_v3_8(metadata) + expected_metadata = {"sampling_frequency": 3.90625, "num_channels": 1, "num_planes": 1} + assert metadata == expected_metadata + + +@pytest.mark.parametrize( + "filename, expected_timestamps", + [ + ("scanimage_20220801_single.tif", [0.45951611, 0.98468446, 1.50985974]), + ( + "scanimage_20220923_roi.tif", + [ + 0.0, + 0.0, + 0.03433645, + 0.03433645, + 1.04890375, + 1.04890375, + 1.08324025, + 1.08324025, + 2.12027815, + 2.12027815, + 2.15461465, + 2.15461465, + 2.7413649, + 2.7413649, + 2.7757014, + 2.7757014, + 3.23987545, + 3.23987545, + 3.27421195, + 3.27421195, + 3.844804, + 3.844804, + 3.87914055, + 3.87914055, + ], + ), + ], +) +def test_extract_timestamps_from_file(filename, expected_timestamps): + file_path = OPHYS_DATA_PATH / "imaging_datasets" / "ScanImage" / filename + timestamps = extract_timestamps_from_file(file_path) + assert_array_equal(timestamps, expected_timestamps) From aac88dd0baef508bb732051116fe8da75fd9a4f9 Mon Sep 17 00:00:00 2001 From: pauladkisson Date: Thu, 28 Sep 2023 17:27:20 -0700 Subject: [PATCH 10/16] switched from local MultiPlaneIE to rebased VolumetricIE --- .../scanimagetiffimagingextractor.py | 146 +----------------- 1 file changed, 2 insertions(+), 144 deletions(-) diff --git a/src/roiextractors/extractors/tiffimagingextractors/scanimagetiffimagingextractor.py b/src/roiextractors/extractors/tiffimagingextractors/scanimagetiffimagingextractor.py index 0ad12b43..db0e3209 100644 --- a/src/roiextractors/extractors/tiffimagingextractors/scanimagetiffimagingextractor.py +++ b/src/roiextractors/extractors/tiffimagingextractors/scanimagetiffimagingextractor.py @@ -12,6 +12,7 @@ from ...extraction_tools import PathType, FloatType, ArrayType, DtypeType, get_package from ...imagingextractor import ImagingExtractor +from ...volumetricimagingextractor import VolumetricImagingExtractor from .scanimagetiff_utils import ( extract_extra_metadata, parse_metadata, @@ -20,150 +21,7 @@ ) -class MultiPlaneImagingExtractor(ImagingExtractor): - """Class to combine multiple ImagingExtractor objects by depth plane.""" - - extractor_name = "MultiPlaneImaging" - installed = True - installatiuon_mesage = "" - - def __init__(self, imaging_extractors: List[ImagingExtractor]): - """Initialize a MultiPlaneImagingExtractor object from a list of ImagingExtractors. - - Parameters - ---------- - imaging_extractors: list of ImagingExtractor - list of imaging extractor objects - """ - super().__init__() - assert isinstance(imaging_extractors, list), "Enter a list of ImagingExtractor objects as argument" - assert all(isinstance(imaging_extractor, ImagingExtractor) for imaging_extractor in imaging_extractors) - self._check_consistency_between_imaging_extractors(imaging_extractors) - self._imaging_extractors = imaging_extractors - self._num_planes = len(imaging_extractors) - - # TODO: Add consistency check for channel_names when API is standardized - def _check_consistency_between_imaging_extractors(self, imaging_extractors: List[ImagingExtractor]): - """Check that essential properties are consistent between extractors so that they can be combined appropriately. - - Parameters - ---------- - imaging_extractors: list of ImagingExtractor - list of imaging extractor objects - - Raises - ------ - AssertionError - If any of the properties are not consistent between extractors. - - Notes - ----- - This method checks the following properties: - - sampling frequency - - image size - - number of channels - - channel names - - data type - - num_frames - """ - properties_to_check = dict( - get_sampling_frequency="The sampling frequency", - get_image_size="The size of a frame", - get_num_channels="The number of channels", - get_dtype="The data type", - get_num_frames="The number of frames", - ) - for method, property_message in properties_to_check.items(): - values = [getattr(extractor, method)() for extractor in imaging_extractors] - unique_values = set(tuple(v) if isinstance(v, Iterable) else v for v in values) - assert ( - len(unique_values) == 1 - ), f"{property_message} is not consistent over the files (found {unique_values})." - - def get_video(self, start_frame: Optional[int] = None, end_frame: Optional[int] = None) -> np.ndarray: - """Get the video frames. - - Parameters - ---------- - start_frame: int, optional - Start frame index (inclusive). - end_frame: int, optional - End frame index (exclusive). - - Returns - ------- - video: numpy.ndarray - The 3D video frames (num_rows, num_columns, num_planes). - """ - start_frame = start_frame if start_frame is not None else 0 - end_frame = end_frame if end_frame is not None else self.get_num_frames() - - video = np.zeros((end_frame - start_frame, *self.get_image_size()), self.get_dtype()) - for i, imaging_extractor in enumerate(self._imaging_extractors): - video[..., i] = imaging_extractor.get_video(start_frame, end_frame) - return video - - def get_frames(self, frame_idxs: ArrayType) -> np.ndarray: - """Get specific video frames from indices (not necessarily continuous). - - Parameters - ---------- - frame_idxs: array-like - Indices of frames to return. - - Returns - ------- - frames: numpy.ndarray - The 3D video frames (num_rows, num_columns, num_planes). - """ - if isinstance(frame_idxs, int): - frame_idxs = [frame_idxs] - - if not all(np.diff(frame_idxs) == 1): - frames = np.zeros((len(frame_idxs), *self.get_image_size()), self.get_dtype()) - for i, imaging_extractor in enumerate(self._imaging_extractors): - frames[..., i] = imaging_extractor.get_frames(frame_idxs) - else: - return self.get_video(start_frame=frame_idxs[0], end_frame=frame_idxs[-1] + 1) - - def get_image_size(self) -> Tuple: - """Get the size of a single frame. - - Returns - ------- - image_size: tuple - The size of a single frame (num_rows, num_columns, num_planes). - """ - image_size = (*self._imaging_extractors[0].get_image_size(), self.get_num_planes()) - return image_size - - def get_num_planes(self) -> int: - """Get the number of depth planes. - - Returns - ------- - _num_planes: int - The number of depth planes. - """ - return self._num_planes - - def get_num_frames(self) -> int: - return self._imaging_extractors[0].get_num_frames() - - def get_sampling_frequency(self) -> float: - return self._imaging_extractors[0].get_sampling_frequency() - - def get_channel_names(self) -> list: - return self._imaging_extractors[0].get_channel_names() - - def get_num_channels(self) -> int: - return self._imaging_extractors[0].get_num_channels() - - def get_dtype(self) -> DtypeType: - return self._imaging_extractors[0].get_dtype() - - -class ScanImageTiffMultiPlaneImagingExtractor(MultiPlaneImagingExtractor): +class ScanImageTiffMultiPlaneImagingExtractor(VolumetricImagingExtractor): """Specialized extractor for reading multi-plane (volumetric) TIFF files produced via ScanImage.""" extractor_name = "ScanImageTiffMultiPlaneImaging" From 10bcab0164bf3518147ff67b2cfec124ec5e5120 Mon Sep 17 00:00:00 2001 From: pauladkisson Date: Thu, 28 Sep 2023 17:33:34 -0700 Subject: [PATCH 11/16] renamed class methods to avoid conflicts with old IE's --- .../scanimagetiffimagingextractor.py | 11 +++++++---- tests/test_scanimagetiffimagingextractor.py | 4 ++-- 2 files changed, 9 insertions(+), 6 deletions(-) diff --git a/src/roiextractors/extractors/tiffimagingextractors/scanimagetiffimagingextractor.py b/src/roiextractors/extractors/tiffimagingextractors/scanimagetiffimagingextractor.py index db0e3209..1c755b64 100644 --- a/src/roiextractors/extractors/tiffimagingextractors/scanimagetiffimagingextractor.py +++ b/src/roiextractors/extractors/tiffimagingextractors/scanimagetiffimagingextractor.py @@ -60,8 +60,8 @@ class ScanImageTiffSinglePlaneImagingExtractor(ImagingExtractor): mode = "file" @classmethod - def get_channel_names(cls, file_path): - """Get the channel names from a TIFF file produced by ScanImage. + def get_available_channels(cls, file_path): + """Get the available channel names from a TIFF file produced by ScanImage. Parameters ---------- @@ -79,8 +79,8 @@ def get_channel_names(cls, file_path): return channel_names @classmethod - def get_plane_names(cls, file_path): - """Get the plane names from a TIFF file produced by ScanImage. + def get_available_planes(cls, file_path): + """Get the available plane names from a TIFF file produced by ScanImage. Parameters ---------- @@ -275,6 +275,9 @@ def get_num_frames(self) -> int: def get_sampling_frequency(self) -> float: return self._sampling_frequency + def get_channel_names(self) -> list: + return self._channel_names + def get_num_channels(self) -> int: return self._num_channels diff --git a/tests/test_scanimagetiffimagingextractor.py b/tests/test_scanimagetiffimagingextractor.py index 845fc443..647cbd52 100644 --- a/tests/test_scanimagetiffimagingextractor.py +++ b/tests/test_scanimagetiffimagingextractor.py @@ -166,9 +166,9 @@ def test_get_num_channels(scan_image_tiff_single_plane_imaging_extractor, expect assert num_channels == expected_properties["num_channels"] -def test_get_channel_names(scan_image_tiff_single_plane_imaging_extractor, expected_properties): +def test_get_available_channels(scan_image_tiff_single_plane_imaging_extractor, expected_properties): file_path = str(scan_image_tiff_single_plane_imaging_extractor.file_path) - channel_names = ScanImageTiffSinglePlaneImagingExtractor.get_channel_names(file_path) + channel_names = ScanImageTiffSinglePlaneImagingExtractor.get_available_channels(file_path) assert channel_names == expected_properties["channel_names"] From a44e4aa8814171aac3a4b4ee1677c7dddad9064f Mon Sep 17 00:00:00 2001 From: pauladkisson Date: Thu, 28 Sep 2023 17:40:14 -0700 Subject: [PATCH 12/16] added invalid test for parse_matlab_vector --- tests/test_scanimage_utils.py | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/tests/test_scanimage_utils.py b/tests/test_scanimage_utils.py index 4b65b0cf..cbb2da09 100644 --- a/tests/test_scanimage_utils.py +++ b/tests/test_scanimage_utils.py @@ -47,6 +47,11 @@ def test_parse_matlab_vector(matlab_vector, expected_vector): assert vector == expected_vector +def test_parse_matlab_vector_invalid(): + with pytest.raises(ValueError): + parse_matlab_vector("Invalid vector") + + @pytest.mark.parametrize( "filename, expected_metadata", [ From ead321b9c220187b0a7e2ba2492e81642ecd9d4a Mon Sep 17 00:00:00 2001 From: pauladkisson Date: Thu, 28 Sep 2023 17:43:53 -0700 Subject: [PATCH 13/16] added invalid test for parse_matlab_vector --- tests/test_scanimage_utils.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tests/test_scanimage_utils.py b/tests/test_scanimage_utils.py index cbb2da09..fd55c671 100644 --- a/tests/test_scanimage_utils.py +++ b/tests/test_scanimage_utils.py @@ -49,7 +49,7 @@ def test_parse_matlab_vector(matlab_vector, expected_vector): def test_parse_matlab_vector_invalid(): with pytest.raises(ValueError): - parse_matlab_vector("Invalid vector") + parse_matlab_vector("Invalid") @pytest.mark.parametrize( From dcaa04a2571e82eb7b4bab0226ce3e73637a62f3 Mon Sep 17 00:00:00 2001 From: pauladkisson Date: Fri, 29 Sep 2023 11:58:32 -0700 Subject: [PATCH 14/16] added test for get_available_planes --- tests/test_scanimagetiffimagingextractor.py | 7 +++++++ 1 file changed, 7 insertions(+) diff --git a/tests/test_scanimagetiffimagingextractor.py b/tests/test_scanimagetiffimagingextractor.py index 647cbd52..4c0889a9 100644 --- a/tests/test_scanimagetiffimagingextractor.py +++ b/tests/test_scanimagetiffimagingextractor.py @@ -19,6 +19,7 @@ def expected_properties(): num_planes=2, frames_per_slice=2, channel_names=["Channel 1", "Channel 4"], + plane_names=["0", "1"], image_size=(528, 256), num_frames=6, dtype="int16", @@ -166,6 +167,12 @@ def test_get_num_channels(scan_image_tiff_single_plane_imaging_extractor, expect assert num_channels == expected_properties["num_channels"] +def test_get_available_planes(scan_image_tiff_single_plane_imaging_extractor, expected_properties): + file_path = str(scan_image_tiff_single_plane_imaging_extractor.file_path) + plane_names = ScanImageTiffSinglePlaneImagingExtractor.get_available_planes(file_path) + assert plane_names == expected_properties["plane_names"] + + def test_get_available_channels(scan_image_tiff_single_plane_imaging_extractor, expected_properties): file_path = str(scan_image_tiff_single_plane_imaging_extractor.file_path) channel_names = ScanImageTiffSinglePlaneImagingExtractor.get_available_channels(file_path) From 5cdd76cde3bd4e9d4b4b648985b61780bda5fb86 Mon Sep 17 00:00:00 2001 From: pauladkisson Date: Fri, 29 Sep 2023 12:02:57 -0700 Subject: [PATCH 15/16] removed unnecessary (and unreachable) suffix check/warning --- .../tiffimagingextractors/scanimagetiffimagingextractor.py | 7 ------- 1 file changed, 7 deletions(-) diff --git a/src/roiextractors/extractors/tiffimagingextractors/scanimagetiffimagingextractor.py b/src/roiextractors/extractors/tiffimagingextractors/scanimagetiffimagingextractor.py index 1c755b64..fc260d84 100644 --- a/src/roiextractors/extractors/tiffimagingextractors/scanimagetiffimagingextractor.py +++ b/src/roiextractors/extractors/tiffimagingextractors/scanimagetiffimagingextractor.py @@ -147,13 +147,6 @@ def __init__( raise ValueError(f"Plane name ({plane_name}) not found in plane names ({self._plane_names}).") self.plane = self._plane_names.index(plane_name) - valid_suffixes = [".tiff", ".tif", ".TIFF", ".TIF"] - if self.file_path.suffix not in valid_suffixes: - suffix_string = ", ".join(valid_suffixes[:-1]) + f", or {valid_suffixes[-1]}" - warn( - f"Suffix ({self.file_path.suffix}) is not of type {suffix_string}! " - f"The {self.extractor_name}Extractor may not be appropriate for the file." - ) ScanImageTiffReader = _get_scanimage_reader() with ScanImageTiffReader(str(self.file_path)) as io: shape = io.shape() # [frames, rows, columns] From 2d4a19bef3fc3672e66138ef24ce9842f32a11e9 Mon Sep 17 00:00:00 2001 From: pauladkisson Date: Wed, 25 Oct 2023 13:38:26 -0700 Subject: [PATCH 16/16] added missing docstrings --- .../scanimagetiff_utils.py | 2 +- .../scanimagetiffimagingextractor.py | 19 +++++++++++++++++++ 2 files changed, 20 insertions(+), 1 deletion(-) diff --git a/src/roiextractors/extractors/tiffimagingextractors/scanimagetiff_utils.py b/src/roiextractors/extractors/tiffimagingextractors/scanimagetiff_utils.py index 49f1c981..691722dc 100644 --- a/src/roiextractors/extractors/tiffimagingextractors/scanimagetiff_utils.py +++ b/src/roiextractors/extractors/tiffimagingextractors/scanimagetiff_utils.py @@ -1,5 +1,5 @@ +"""Utility functions for ScanImage TIFF Extractors.""" import numpy as np - from ...extraction_tools import PathType, get_package diff --git a/src/roiextractors/extractors/tiffimagingextractors/scanimagetiffimagingextractor.py b/src/roiextractors/extractors/tiffimagingextractors/scanimagetiffimagingextractor.py index fc260d84..fea05b62 100644 --- a/src/roiextractors/extractors/tiffimagingextractors/scanimagetiffimagingextractor.py +++ b/src/roiextractors/extractors/tiffimagingextractors/scanimagetiffimagingextractor.py @@ -275,12 +275,31 @@ def get_num_channels(self) -> int: return self._num_channels def get_num_planes(self) -> int: + """Get the number of depth planes. + + Returns + ------- + _num_planes: int + The number of depth planes. + """ return self._num_planes def get_dtype(self) -> DtypeType: return self.get_frames(0).dtype def check_frame_inputs(self, frame) -> None: + """Check that the frame index is valid. Raise ValueError if not. + + Parameters + ---------- + frame : int + The index of the frame to retrieve. + + Raises + ------ + ValueError + If the frame index is invalid. + """ if frame >= self._num_frames: raise ValueError(f"Frame index ({frame}) exceeds number of frames ({self._num_frames}).") if frame < 0: