forked from pydicom/pydicom
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathdicomdir.py
87 lines (76 loc) · 3.83 KB
/
dicomdir.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
# dicomdir.py
"""Module for DicomDir class"""
#
# Copyright (c) 2013 Darcy Mason
# This file is part of pydicom, released under a modified MIT license.
# See the file license.txt included with this distribution, also
# available at https://github.com/darcymason/pydicom
#
from pydicom.errors import InvalidDicomError
from pydicom.dataset import FileDataset
class DicomDir(FileDataset):
"""Hold a DICOMDIR dataset read from file.
Derived from FileDataset, but additional methods are available,
specific to the Directory structure
"""
def __init__(self, filename_or_obj, dataset, preamble=None, file_meta=None,
is_implicit_VR=True, is_little_endian=True):
"""Initialize a DICOMDIR dataset read from a DICOM file
Carries forward all the initialization from FileDataset class
:param filename: full path and filename to the file. Use None if is a BytesIO.
:param dataset: some form of dictionary, usually a Dataset from read_dataset()
:param preamble: the 128-byte DICOM preamble
:param file_meta: the file meta info dataset, as returned by _read_file_meta,
or an empty dataset if no file meta information is in the file
:param is_implicit_VR: True if implicit VR transfer syntax used; False if explicit VR. Default is True.
:param is_little_endian: True if little-endian transfer syntax used; False if big-endian. Default is True.
"""
# Usually this class is created through filereader.read_partial,
# and it checks class SOP, but in case of direct creation,
# check here also
if file_meta:
class_uid = file_meta.MediaStorageSOPClassUID
if not class_uid == "Media Storage Directory Storage":
msg = "SOP Class is not Media Storage Directory (DICOMDIR)"
raise InvalidDicomError(msg)
FileDataset.__init__(self, filename_or_obj, dataset,
preamble, file_meta,
is_implicit_VR=True, is_little_endian=True)
self.parse_records()
def parse_records(self):
"""Build the hierarchy of given directory records, and structure
into Patient, Studies, Series, Images hierarchy.
This is intended for initial read of file only,
it will not reorganize correctly if records are changed.
:return: None
"""
# Define a helper function for organizing the records
def get_siblings(record, map_offset_to_record):
"""Return a list of all siblings of the given directory record,
including itself.
"""
sibling_list = [record]
current_record = record
while current_record.OffsetOfTheNextDirectoryRecord:
offset_of_next = current_record.OffsetoftheNextDirectoryRecord
sibling = map_offset_to_record[offset_of_next]
sibling_list.append(sibling)
current_record = sibling
return sibling_list
# Build the mapping from file offsets to records
records = self.DirectoryRecordSequence
map_offset_to_record = {}
for record in records:
offset = record.seq_item_tell
map_offset_to_record[offset] = record
# logging.debug("Record offsets: " + map_offset_to_record.keys())
# Find the children of each record
for record in records:
child_offset = record.OffsetOfReferencedLowerLevelDirectoryEntity
if child_offset:
child = map_offset_to_record[child_offset]
record.children = get_siblings(child, map_offset_to_record)
else:
record.children = []
# Find the top-level records : siblings of the first record
self.patient_records = get_siblings(records[0], map_offset_to_record)