Skip to content

Commit

Permalink
Merge pull request #124 from sustrev/main
Browse files Browse the repository at this point in the history
Can now save .fda/.fds as full DICOM files, and improves to their readers + metadata extraction.
  • Loading branch information
marksgraham authored Sep 11, 2023
2 parents 897d72f + 9e011df commit 9cf5baf
Show file tree
Hide file tree
Showing 13 changed files with 1,520 additions and 198 deletions.
8 changes: 8 additions & 0 deletions examples/demo_fda_extraction.py
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
import json

from oct_converter.dicom import create_dicom_from_oct
from oct_converter.readers import FDA

# a sample .fda file can be downloaded from the Biobank resource here:
Expand Down Expand Up @@ -32,3 +33,10 @@
metadata = fda.read_all_metadata()
with open("metadata.json", "w") as outfile:
outfile.write(json.dumps(metadata, indent=4))

# create a DICOM from FDA
dcm = create_dicom_from_oct(filepath)
# Output dir can be specified, otherwise will
# default to current working directory.
# Output filename can be specified, otherwise
# will default to the input filename.
8 changes: 8 additions & 0 deletions examples/demo_fds_extraction.py
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
import json

from oct_converter.dicom import create_dicom_from_oct
from oct_converter.readers import FDS

# An example .fds file can be downloaded from the Biobank website:
Expand All @@ -26,3 +27,10 @@
metadata = fds.read_all_metadata(verbose=True)
with open("fds_metadata.json", "w") as outfile:
outfile.write(json.dumps(metadata, indent=4))

# create a DICOM from FDS
dcm = create_dicom_from_oct(filepath)
# Output dir can be specified, otherwise will
# default to current working directory.
# Output filename can be specified, otherwise
# will default to the input filename.
3 changes: 3 additions & 0 deletions oct_converter/dicom/__init__.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
"""Init module."""

from .dicom import create_dicom_from_oct
295 changes: 295 additions & 0 deletions oct_converter/dicom/dicom.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,295 @@
import typing as t
from datetime import datetime
from importlib import metadata
from pathlib import Path

import numpy as np
from pydicom.dataset import Dataset, FileDataset, FileMetaDataset
from pydicom.uid import (
ExplicitVRLittleEndian,
OphthalmicTomographyImageStorage,
generate_uid,
)

from oct_converter.dicom.fda_meta import fda_dicom_metadata
from oct_converter.dicom.fds_meta import fds_dicom_metadata
from oct_converter.dicom.metadata import DicomMetadata
from oct_converter.readers import FDA, FDS

# Deterministic implentation UID based on package name and version
version = metadata.version("oct_converter")
implementation_uid = generate_uid(entropy_srcs=["oct_converter", version])


def opt_base_dicom(filepath: Path) -> Dataset:
"""Creates the base dicom to be populated.
Args:
filepath: Path to where output file is to be saved
Returns:
ds: FileDataset with file meta, preamble, and empty dataset
"""
# Populate required values for file meta information
file_meta = FileMetaDataset()
file_meta.MediaStorageSOPClassUID = OphthalmicTomographyImageStorage
file_meta.MediaStorageSOPInstanceUID = generate_uid()
file_meta.ImplementationClassUID = implementation_uid
file_meta.TransferSyntaxUID = ExplicitVRLittleEndian

# Create the FileDataset instance with file meta, preamble and empty DS
ds = FileDataset(str(filepath), {}, file_meta=file_meta, preamble=b"\0" * 128)
ds.is_little_endian = True
ds.is_implicit_VR = False # Explicit VR
return ds


def populate_patient_info(ds: Dataset, meta: DicomMetadata) -> Dataset:
"""Populates Patient Module PS3.3 C.7.1.1
Args:
ds: current dataset
meta: DICOM metadata information
Returns:
ds: Dataset, updated with patient information
"""
# Patient Module PS3.3 C.7.1.1
ds.PatientName = f"{meta.patient_info.last_name}^{meta.patient_info.first_name}"
ds.PatientID = meta.patient_info.patient_id
ds.PatientSex = meta.patient_info.patient_sex
ds.PatientBirthDate = (
meta.patient_info.patient_dob.strftime("%Y%m%d")
if meta.patient_info.patient_dob
else ""
)
return ds


def populate_manufacturer_info(ds: Dataset, meta: DicomMetadata) -> Dataset:
"""Populates equipment modules PS3.3 C.7.5.1, PS3.3 C.7.5.2
Args:
ds: current dataset
meta: DICOM metadata information
Returns:
ds: Dataset, updated with equipment information
"""
# General and enhanced equipment module PS3.3 C.7.5.1, PS3.3 C.7.5.2
ds.Manufacturer = meta.manufacturer_info.manufacturer
ds.ManufacturerModelName = meta.manufacturer_info.manufacturer_model
ds.DeviceSerialNumber = meta.manufacturer_info.device_serial
ds.SoftwareVersions = meta.manufacturer_info.software_version

# OPT parameter module PS3.3 C.8.17.9
cd, cv, cm = meta.oct_image_params.opt_acquisition_device.value
ds.AcquisitionDeviceTypeCodeSequence = [Dataset()]
ds.AcquisitionDeviceTypeCodeSequence[0].CodeValue = cv
ds.AcquisitionDeviceTypeCodeSequence[0].CodingSchemeDesignator = cd
ds.AcquisitionDeviceTypeCodeSequence[0].CodeMeaning = cm
ds.DetectorType = meta.oct_image_params.DetectorType.value
return ds


def populate_opt_series(ds: Dataset, meta: DicomMetadata) -> Dataset:
"""Populates study and series modules, PS3.3 C.7.2.1, PS3.3 C.7.3.1,
PS3.3 C.8.17.6, and PS3.3 C.12.1
Args:
ds: current dataset
meta: DICOM metadata information
Returns:
ds: Dataset, updated with study and series information
"""
# General study module PS3.3 C.7.2.1
# Deterministic StudyInstanceUID based on study ID
# ds.StudyInstanceUID = generate_uid(entropy_srcs=[
# # str(uuid.uuid4()),
# str(meta.series_info.study_id)
# ])

# # General series module PS3.3 C.7.3.1
# ds.SeriesInstanceUID = generate_uid(entropy_srcs=[
# # str(uuid.uuid4()),
# str(meta.series_info.series_id)
# ])
ds.StudyInstanceUID = generate_uid()
ds.SeriesInstanceUID = generate_uid()
ds.Laterality = meta.series_info.laterality
# Ophthalmic Tomography Series PS3.3 C.8.17.6
ds.Modality = "OPT"
ds.SeriesNumber = int(meta.series_info.series_id)

# SOP Common module PS3.3 C.12.1
ds.SOPClassUID = OphthalmicTomographyImageStorage
ds.SOPInstanceUID = generate_uid()
return ds


def populate_ocular_region(ds: Dataset, meta: DicomMetadata) -> Dataset:
"""Populates ocular region modules, PS3.3 C.8.17.5, PS3.3 C.7.6.16.2.8,
and PS3.3 C.7.6.16.2.1
Args:
ds: current dataset
meta: DICOM metadata information
Returns:
ds: Dataset, updated with ocular region information
"""
# Ocular region imaged module PS3.3 C.8.17.5
cd, cv, cm = meta.series_info.opt_anatomy.value
ds.ImageLaterality = meta.series_info.laterality
ds.AnatomicRegionSequence = [Dataset()]
ds.AnatomicRegionSequence[0].CodeValue = cv
ds.AnatomicRegionSequence[0].CodingSchemeDesignator = cd
ds.AnatomicRegionSequence[0].CodeMeaning = cm
return ds


def opt_shared_functional_groups(ds: Dataset, meta: DicomMetadata) -> Dataset:
# ---- Shared
shared_ds = [Dataset()]
# Frame anatomy PS3.3 C.7.6.16.2.8
shared_ds[0].FrameAnatomySequence = [Dataset()]
shared_ds[0].FrameAnatomySequence[0] = ds.AnatomicRegionSequence[0].copy()
shared_ds[0].FrameAnatomySequence[0].FrameLaterality = meta.series_info.laterality
# Pixel Measures PS3.3 C.7.6.16.2.1
shared_ds[0].PixelMeasuresSequence = [Dataset()]
shared_ds[0].PixelMeasuresSequence[
0
].PixelSpacing = meta.image_geometry.pixel_spacing
shared_ds[0].PixelMeasuresSequence[
0
].SliceThickness = meta.image_geometry.slice_thickness
# Plane Orientation PS3.3 C.7.6.16.2.4
shared_ds[0].PlaneOrientationSequence = [Dataset()]
shared_ds[0].PlaneOrientationSequence[
0
].ImageOrientationPatient = meta.image_geometry.image_orientation
ds.SharedFunctionalGroupsSequence = shared_ds
return ds


def write_opt_dicom(
meta: DicomMetadata, frames: t.List[np.ndarray], filepath: Path
) -> Path:
"""Writes required DICOM metadata and pixel data to .dcm file.
Args:
meta: DICOM metadata information
frames: list of frames of pixel data
filepath: Path to where output file is being saved
Returns:
Path to created DICOM file
"""
ds = opt_base_dicom(filepath)
ds = populate_patient_info(ds, meta)
ds = populate_manufacturer_info(ds, meta)
ds = populate_opt_series(ds, meta)
ds = populate_ocular_region(ds, meta)
ds = opt_shared_functional_groups(ds, meta)

# TODO: Frame of reference if fundus image present

# OPT Image Module PS3.3 C.8.17.7
ds.ImageType = ["DERIVED", "SECONDARY"]
ds.SamplesPerPixel = 1
ds.AcquisitionDateTime = (
meta.series_info.acquisition_date.strftime("%Y%m%d%H%M%S.%f")
if meta.series_info.acquisition_date
else ""
)
ds.AcquisitionNumber = 1
ds.PhotometricInterpretation = "MONOCHROME2"
# Unsigned integer
ds.PixelRepresentation = 0
# Use 16 bit pixel
ds.BitsAllocated = 16
ds.BitsStored = ds.BitsAllocated
ds.HighBit = ds.BitsAllocated - 1
ds.SamplesPerPixel = 1
ds.NumberOfFrames = len(frames)

# Multi-frame Functional Groups Module PS3.3 C.7.6.16
dt = datetime.now()
ds.ContentDate = dt.strftime("%Y%m%d")
timeStr = dt.strftime("%H%M%S.%f") # long format with micro seconds
ds.ContentTime = timeStr
ds.InstanceNumber = 1

per_frame = []
pixel_data_bytes = list()
# Convert to a 3d volume
pixel_data = np.array(frames).astype(np.uint16)
ds.Rows = pixel_data.shape[1]
ds.Columns = pixel_data.shape[2]
for i in range(pixel_data.shape[0]):
# Per Frame Functional Groups
frame_fgs = Dataset()
frame_fgs.PlanePositionSequence = [Dataset()]
ipp = [0, 0, i * meta.image_geometry.slice_thickness]
frame_fgs.PlanePositionSequence[0].ImagePositionPatient = ipp
frame_fgs.FrameContentSequence = [Dataset()]
frame_fgs.FrameContentSequence[0].InStackPositionNumber = i + 1
frame_fgs.FrameContentSequence[0].StackID = "1"

# Pixel data
frame_dat = pixel_data[i, :, :]
pixel_data_bytes.append(frame_dat.tobytes())
per_frame.append(frame_fgs)
ds.PerFrameFunctionalGroupsSequence = per_frame
ds.PixelData = pixel_data.tobytes()
ds.save_as(filepath)
return filepath


def create_dicom_from_oct(
input_file: str, output_dir: str = None, output_filename: str = None
) -> Path:
"""Creates a DICOM file with the data parsed from
the input file.
Args:
input_file: File with OCT data (Currently only Topcon
files supported)
output_dir: Output directory, will be created if
not currently exists. Default None places file in
current working directory.
output_filename: Name to save the file under, i.e.
`filename.dcm`. Default None saves the file under
the input filename (if input_file = `test.fds`,
output_filename = `test.dcm`)
Returns:
Path to DICOM file
"""
file_suffix = input_file.split(".")[-1].lower()
if file_suffix == "fds":
fds = FDS(input_file)
oct = fds.read_oct_volume()
meta = fds_dicom_metadata(oct)
elif file_suffix == "fda":
fda = FDA(input_file)
oct = fda.read_oct_volume()
meta = fda_dicom_metadata(oct)
elif file_suffix in ["e2e", "img", "oct"]:
raise NotImplementedError(
f"DICOM conversion for {file_suffix} is not yet supported. Currently supported filetypes are .fds, .fda."
)
else:
raise TypeError(
f"DICOM conversion for {file_suffix} is not supported. Currently supported filetypes are .fds, .fda."
)

if output_dir:
output_dir = Path(output_dir)
output_dir.mkdir(parents=True, exist_ok=True)
else:
output_dir = Path.cwd()

if not output_filename:
output_filename = Path(input_file).stem + ".dcm"

filepath = Path(output_dir, output_filename)

file = write_opt_dicom(meta, oct.volume, filepath)

return file
Loading

0 comments on commit 9cf5baf

Please sign in to comment.