SimpleITK  2.0.0
DicomConvert/DicomConvert.py
1 # =========================================================================
2 #
3 # Copyright NumFOCUS
4 #
5 # Licensed under the Apache License, Version 2.0 (the "License");
6 # you may not use this file except in compliance with the License.
7 # You may obtain a copy of the License at
8 #
9 # http://www.apache.org/licenses/LICENSE-2.0.txt
10 #
11 # Unless required by applicable law or agreed to in writing, software
12 # distributed under the License is distributed on an "AS IS" BASIS,
13 # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14 # See the License for the specific language governing permissions and
15 # limitations under the License.
16 #
17 # =========================================================================
18 
19 import argparse
20 import csv
21 import functools
22 import itertools
23 import multiprocessing
24 import os
25 import sys
26 
27 import SimpleITK as sitk
28 
29 
30 def convert_image(input_file_name, output_file_name, new_width=None):
31  try:
32  image_file_reader = sitk.ImageFileReader()
33  # only read DICOM images
34  image_file_reader.SetImageIO('GDCMImageIO')
35  image_file_reader.SetFileName(input_file_name)
36  image_file_reader.ReadImageInformation()
37  image_size = list(image_file_reader.GetSize())
38  if len(image_size) == 3 and image_size[2] == 1:
39  image_size[2] = 0
40  image_file_reader.SetExtractSize(image_size)
41  image = image_file_reader.Execute()
42  if new_width:
43  original_size = image.GetSize()
44  original_spacing = image.GetSpacing()
45  new_spacing = [(original_size[0] - 1) * original_spacing[0]
46  / (new_width - 1)] * 2
47  new_size = [new_width, int((original_size[1] - 1)
48  * original_spacing[1] / new_spacing[1])]
49  image = sitk.Resample(image1=image, size=new_size,
50  transform=sitk.Transform(),
51  interpolator=sitk.sitkLinear,
52  outputOrigin=image.GetOrigin(),
53  outputSpacing=new_spacing,
54  outputDirection=image.GetDirection(),
55  defaultPixelValue=0,
56  outputPixelType=image.GetPixelID())
57  # If a single channel image, rescale to [0,255]. Also modify the
58  # intensity values based on the photometric interpretation. If
59  # MONOCHROME2 (minimum should be displayed as black) we don't need to
60  # do anything, if image has MONOCRHOME1 (minimum should be displayed as
61  # white) we flip # the intensities. This is a constraint imposed by ITK
62  # which always assumes MONOCHROME2.
63  if image.GetNumberOfComponentsPerPixel() == 1:
64  image = sitk.RescaleIntensity(image, 0, 255)
65  if image_file_reader.GetMetaData('0028|0004').strip() \
66  == 'MONOCHROME1':
67  image = sitk.InvertIntensity(image, maximum=255)
68  image = sitk.Cast(image, sitk.sitkUInt8)
69  sitk.WriteImage(image, output_file_name)
70  return True
71  except BaseException:
72  return False
73 
74 
75 def convert_images(input_file_names, output_file_names, new_width):
76  MAX_PROCESSES = 15
77  with multiprocessing.Pool(processes=MAX_PROCESSES) as pool:
78  return pool.starmap(functools.partial(convert_image,
79  new_width=new_width),
80  zip(input_file_names, output_file_names))
81 
82 
83 def positive_int(int_str):
84  value = int(int_str)
85  if value <= 0:
86  raise argparse.ArgumentTypeError(
87  int_str + ' is not a positive integer value')
88  return value
89 
90 
91 def directory(dir_name):
92  if not os.path.isdir(dir_name):
93  raise argparse.ArgumentTypeError(dir_name +
94  ' is not a valid directory name')
95  return dir_name
96 
97 
98 def main(argv=None):
99  parser = argparse.ArgumentParser(
100  description='Convert and resize DICOM files to common image types.')
101  parser.add_argument('root_of_data_directory', type=directory,
102  help='Path to the topmost directory containing data.')
103  parser.add_argument(
104  'output_file_extension',
105  help='Image file extension, this determines output file type '
106  '(e.g. png) .')
107  parser.add_argument('--w', type=positive_int,
108  help='Width of converted images.')
109  parser.add_argument('--od', type=directory, help='Output directory.')
110  args = parser.parse_args(argv)
111 
112  input_file_names = []
113  for dir_name, subdir_names, file_names in os.walk(
114  args.root_of_data_directory):
115  input_file_names += [os.path.join(os.path.abspath(dir_name), fname)
116  for fname in file_names]
117  if args.od:
118  # if all output files are written to the same directory we need them
119  # to have a unique name, so use an index.
120  file_names = [os.path.join(os.path.abspath(args.od), str(i))
121  for i in range(len(input_file_names))]
122  else:
123  file_names = input_file_names
124  output_file_names = [file_name + '.' + args.output_file_extension
125  for file_name in file_names]
126 
127  res = convert_images(input_file_names, output_file_names, args.w)
128  input_file_names = list(itertools.compress(input_file_names, res))
129  output_file_names = list(itertools.compress(output_file_names, res))
130 
131  # save csv file mapping input and output file names.
132  # using csv module and not pandas so as not to create more dependencies
133  # for the examples. pandas based code is more elegant/shorter.
134  dir_name = args.od if args.od else os.getcwd()
135  with open(os.path.join(dir_name, 'file_names.csv'), mode='w') as fp:
136  fp_writer = csv.writer(fp, delimiter=',', quotechar='"',
137  quoting=csv.QUOTE_MINIMAL)
138  fp_writer.writerow(['input file name', 'output file name'])
139  for data in zip(input_file_names, output_file_names):
140  fp_writer.writerow(data)
141 
142 
143 if __name__ == "__main__":
144  sys.exit(main())
itk::simple::Transform
A simplified wrapper around a variety of ITK transforms.
Definition: sitkTransform.h:80
itk::simple::RescaleIntensity
Image RescaleIntensity(const Image &image1, double outputMinimum=0, double outputMaximum=255)
Applies a linear transformation to the intensity levels of the input Image .
itk::simple::WriteImage
SITKIO_EXPORT void WriteImage(const Image &image, const std::vector< std::string > &fileNames, bool useCompression=false, int compressionLevel=-1)
WriteImage is a procedural interface to the ImageSeriesWriter. class which is convenient for many ima...
itk::simple::InvertIntensity
Image InvertIntensity(const Image &image1, double maximum=255)
Invert the intensity of an image.
itk::simple::ImageFileReader
Read an image file and return a SimpleITK Image.
Definition: sitkImageFileReader.h:60
itk::simple::Cast
Image Cast(const Image &image, PixelIDValueEnum pixelID)
itk::simple::Resample
Image Resample(const Image &image1, const std::vector< uint32_t > &size, Transform transform=itk::simple::Transform(), InterpolatorEnum interpolator=itk::simple::sitkLinear, const std::vector< double > &outputOrigin=std::vector< double >(3, 0.0), const std::vector< double > &outputSpacing=std::vector< double >(3, 1.0), const std::vector< double > &outputDirection=std::vector< double >(), double defaultPixelValue=0.0, PixelIDValueEnum outputPixelType=sitkUnknown, bool useNearestNeighborExtrapolator=false)
itk::simple::ResampleImageFilter Procedural Interface