dicom 0.9.6 → 0.9.7
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +5 -13
- data/CHANGELOG.md +390 -376
- data/COPYING +674 -674
- data/Gemfile +2 -2
- data/Gemfile.lock +30 -28
- data/README.md +154 -152
- data/dicom.gemspec +30 -30
- data/lib/dicom/anonymizer.rb +677 -654
- data/lib/dicom/audit_trail.rb +109 -109
- data/lib/dicom/d_library.rb +269 -265
- data/lib/dicom/d_object.rb +465 -465
- data/lib/dicom/d_read.rb +21 -8
- data/lib/dicom/d_server.rb +329 -329
- data/lib/dicom/d_write.rb +355 -355
- data/lib/dicom/dictionary/elements.tsv +597 -86
- data/lib/dicom/dictionary/uids.tsv +4 -2
- data/lib/dicom/elemental_parent.rb +63 -63
- data/lib/dicom/extensions/array.rb +56 -56
- data/lib/dicom/extensions/hash.rb +30 -30
- data/lib/dicom/extensions/string.rb +125 -125
- data/lib/dicom/file_handler.rb +121 -121
- data/lib/dicom/general/constants.rb +210 -210
- data/lib/dicom/general/deprecated.rb +0 -320
- data/lib/dicom/general/logging.rb +155 -155
- data/lib/dicom/general/methods.rb +98 -82
- data/lib/dicom/general/variables.rb +28 -28
- data/lib/dicom/general/version.rb +5 -5
- data/lib/dicom/image_item.rb +836 -836
- data/lib/dicom/image_processor.rb +79 -79
- data/lib/dicom/image_processor_mini_magick.rb +71 -71
- data/lib/dicom/image_processor_r_magick.rb +106 -106
- data/lib/dicom/link.rb +1529 -1528
- data/rakefile.rb +29 -30
- metadata +43 -49
data/lib/dicom/file_handler.rb
CHANGED
@@ -1,122 +1,122 @@
|
|
1
|
-
module DICOM
|
2
|
-
|
3
|
-
# This class handles DICOM files that have been received through network communication.
|
4
|
-
#
|
5
|
-
# === Notes
|
6
|
-
#
|
7
|
-
# The purpose of this class is to make it as easy as possible for users to customize the way
|
8
|
-
# DICOM files are handled when they are received through the network.
|
9
|
-
#
|
10
|
-
# The default behaviour is to save the files to disk using a folder structure determined by a few select tags of the DICOM file.
|
11
|
-
#
|
12
|
-
# Some suggested alternatives for user customization:
|
13
|
-
# * Analyzing tags and/or image data to determine further actions.
|
14
|
-
# * Modify the DICOM object before it is saved to disk.
|
15
|
-
# * Modify the folder structure in which DICOM files are saved to disk.
|
16
|
-
# * Store DICOM contents in a database (highly relevant if you are building a Ruby on Rails DICOM application).
|
17
|
-
# * Retransmit the DICOM object to another network destination using the DClient class.
|
18
|
-
# * Write information to a log file.
|
19
|
-
#
|
20
|
-
class FileHandler
|
21
|
-
|
22
|
-
# Saves a single DICOM object to file.
|
23
|
-
# Returns a status message stating where the file has been saved.
|
24
|
-
#
|
25
|
-
# Modify this method if you want to change the way your server saves incoming files.
|
26
|
-
#
|
27
|
-
# === Notes
|
28
|
-
#
|
29
|
-
# As default, files will be saved with the following path:
|
30
|
-
# <tt> path_prefix/<PatientID>/<StudyDate>/<Modality>/ </tt>
|
31
|
-
#
|
32
|
-
# === Parameters
|
33
|
-
#
|
34
|
-
# * <tt>path_prefix</tt> -- String. Specifies the root path of the DICOM storage.
|
35
|
-
# * <tt>dcm</tt> -- A DObject instance which will be written to file.
|
36
|
-
# * <tt>transfer_syntax</tt> -- String. Specifies the transfer syntax that will be used to write the DICOM file.
|
37
|
-
#
|
38
|
-
def self.save_file(path_prefix, dcm, transfer_syntax)
|
39
|
-
# File name is set using the SOP Instance UID:
|
40
|
-
file_name = dcm.value("0008,0018") || "missing_SOP_UID"
|
41
|
-
extension = ".dcm"
|
42
|
-
folders = Array.new(3)
|
43
|
-
folders[0] = dcm.value("0010,0020") || "PatientID"
|
44
|
-
folders[1] = dcm.value("0008,0020") || "StudyDate"
|
45
|
-
folders[2] = dcm.value("0008,0060") || "Modality"
|
46
|
-
local_path = folders.join(File::SEPARATOR) + File::SEPARATOR + file_name
|
47
|
-
full_path = path_prefix + local_path + extension
|
48
|
-
# Save the DICOM object to disk:
|
49
|
-
dcm.write(full_path, :transfer_syntax => transfer_syntax)
|
50
|
-
message = [:info, "DICOM file saved to: #{full_path}"]
|
51
|
-
return message
|
52
|
-
end
|
53
|
-
|
54
|
-
# Handles the reception of a series of DICOM objects which are received in a single association.
|
55
|
-
#
|
56
|
-
# Modify this method if you want to change the way your server handles incoming file series.
|
57
|
-
#
|
58
|
-
# === Notes
|
59
|
-
#
|
60
|
-
# Default action: Pass each file to the class method which saves files to disk.
|
61
|
-
#
|
62
|
-
# === Parameters
|
63
|
-
#
|
64
|
-
# * <tt>path</tt> -- String. Specifies the root path of the DICOM storage.
|
65
|
-
# * <tt>objects</tt> -- An array containing the DObject instances which were received.
|
66
|
-
# * <tt>transfer_syntaxes</tt> -- An array containing the transfer syntaxes belonging to the received objects.
|
67
|
-
#
|
68
|
-
def self.receive_files(path, objects, transfer_syntaxes)
|
69
|
-
all_success = true
|
70
|
-
successful, too_short, parse_fail, handle_fail = 0, 0, 0, 0
|
71
|
-
total = objects.length
|
72
|
-
message = nil
|
73
|
-
messages = Array.new
|
74
|
-
# Process each DICOM object:
|
75
|
-
objects.each_index do |i|
|
76
|
-
if objects[i].length > 8
|
77
|
-
# Temporarily increase the log threshold to suppress messages from the DObject class:
|
78
|
-
server_level = DICOM.logger.level
|
79
|
-
DICOM.logger.level = Logger::FATAL
|
80
|
-
# Parse the received data string and load it to a DICOM object:
|
81
|
-
dcm = DObject.parse(objects[i], :no_meta => true, :syntax => transfer_syntaxes[i])
|
82
|
-
# Reset the logg threshold:
|
83
|
-
DICOM.logger.level = server_level
|
84
|
-
if dcm.read?
|
85
|
-
begin
|
86
|
-
message = self.save_file(path, dcm, transfer_syntaxes[i])
|
87
|
-
successful += 1
|
88
|
-
rescue
|
89
|
-
handle_fail += 1
|
90
|
-
all_success = false
|
91
|
-
messages << [:error, "Saving file failed!"]
|
92
|
-
end
|
93
|
-
else
|
94
|
-
parse_fail += 1
|
95
|
-
all_success = false
|
96
|
-
messages << [:error, "Invalid DICOM data encountered: The received string was not parsed successfully."]
|
97
|
-
end
|
98
|
-
else
|
99
|
-
too_short += 1
|
100
|
-
all_success = false
|
101
|
-
messages << [:error, "Invalid data encountered: The received string was too small to contain any DICOM data."]
|
102
|
-
end
|
103
|
-
end
|
104
|
-
# Create a summary status message, when multiple files have been received:
|
105
|
-
if total > 1
|
106
|
-
if successful == total
|
107
|
-
messages << [:info, "All #{total} DICOM files received successfully."]
|
108
|
-
else
|
109
|
-
if successful == 0
|
110
|
-
messages << [:warn, "All #{total} received DICOM files failed!"]
|
111
|
-
else
|
112
|
-
messages << [:warn, "Only #{successful} of #{total} DICOM files received successfully!"]
|
113
|
-
end
|
114
|
-
end
|
115
|
-
else
|
116
|
-
messages = [message] if all_success
|
117
|
-
end
|
118
|
-
return all_success, messages
|
119
|
-
end
|
120
|
-
|
121
|
-
end
|
1
|
+
module DICOM
|
2
|
+
|
3
|
+
# This class handles DICOM files that have been received through network communication.
|
4
|
+
#
|
5
|
+
# === Notes
|
6
|
+
#
|
7
|
+
# The purpose of this class is to make it as easy as possible for users to customize the way
|
8
|
+
# DICOM files are handled when they are received through the network.
|
9
|
+
#
|
10
|
+
# The default behaviour is to save the files to disk using a folder structure determined by a few select tags of the DICOM file.
|
11
|
+
#
|
12
|
+
# Some suggested alternatives for user customization:
|
13
|
+
# * Analyzing tags and/or image data to determine further actions.
|
14
|
+
# * Modify the DICOM object before it is saved to disk.
|
15
|
+
# * Modify the folder structure in which DICOM files are saved to disk.
|
16
|
+
# * Store DICOM contents in a database (highly relevant if you are building a Ruby on Rails DICOM application).
|
17
|
+
# * Retransmit the DICOM object to another network destination using the DClient class.
|
18
|
+
# * Write information to a log file.
|
19
|
+
#
|
20
|
+
class FileHandler
|
21
|
+
|
22
|
+
# Saves a single DICOM object to file.
|
23
|
+
# Returns a status message stating where the file has been saved.
|
24
|
+
#
|
25
|
+
# Modify this method if you want to change the way your server saves incoming files.
|
26
|
+
#
|
27
|
+
# === Notes
|
28
|
+
#
|
29
|
+
# As default, files will be saved with the following path:
|
30
|
+
# <tt> path_prefix/<PatientID>/<StudyDate>/<Modality>/ </tt>
|
31
|
+
#
|
32
|
+
# === Parameters
|
33
|
+
#
|
34
|
+
# * <tt>path_prefix</tt> -- String. Specifies the root path of the DICOM storage.
|
35
|
+
# * <tt>dcm</tt> -- A DObject instance which will be written to file.
|
36
|
+
# * <tt>transfer_syntax</tt> -- String. Specifies the transfer syntax that will be used to write the DICOM file.
|
37
|
+
#
|
38
|
+
def self.save_file(path_prefix, dcm, transfer_syntax)
|
39
|
+
# File name is set using the SOP Instance UID:
|
40
|
+
file_name = dcm.value("0008,0018") || "missing_SOP_UID"
|
41
|
+
extension = ".dcm"
|
42
|
+
folders = Array.new(3)
|
43
|
+
folders[0] = dcm.value("0010,0020") || "PatientID"
|
44
|
+
folders[1] = dcm.value("0008,0020") || "StudyDate"
|
45
|
+
folders[2] = dcm.value("0008,0060") || "Modality"
|
46
|
+
local_path = folders.join(File::SEPARATOR) + File::SEPARATOR + file_name
|
47
|
+
full_path = path_prefix + local_path + extension
|
48
|
+
# Save the DICOM object to disk:
|
49
|
+
dcm.write(full_path, :transfer_syntax => transfer_syntax)
|
50
|
+
message = [:info, "DICOM file saved to: #{full_path}"]
|
51
|
+
return message
|
52
|
+
end
|
53
|
+
|
54
|
+
# Handles the reception of a series of DICOM objects which are received in a single association.
|
55
|
+
#
|
56
|
+
# Modify this method if you want to change the way your server handles incoming file series.
|
57
|
+
#
|
58
|
+
# === Notes
|
59
|
+
#
|
60
|
+
# Default action: Pass each file to the class method which saves files to disk.
|
61
|
+
#
|
62
|
+
# === Parameters
|
63
|
+
#
|
64
|
+
# * <tt>path</tt> -- String. Specifies the root path of the DICOM storage.
|
65
|
+
# * <tt>objects</tt> -- An array containing the DObject instances which were received.
|
66
|
+
# * <tt>transfer_syntaxes</tt> -- An array containing the transfer syntaxes belonging to the received objects.
|
67
|
+
#
|
68
|
+
def self.receive_files(path, objects, transfer_syntaxes)
|
69
|
+
all_success = true
|
70
|
+
successful, too_short, parse_fail, handle_fail = 0, 0, 0, 0
|
71
|
+
total = objects.length
|
72
|
+
message = nil
|
73
|
+
messages = Array.new
|
74
|
+
# Process each DICOM object:
|
75
|
+
objects.each_index do |i|
|
76
|
+
if objects[i].length > 8
|
77
|
+
# Temporarily increase the log threshold to suppress messages from the DObject class:
|
78
|
+
server_level = DICOM.logger.level
|
79
|
+
DICOM.logger.level = Logger::FATAL
|
80
|
+
# Parse the received data string and load it to a DICOM object:
|
81
|
+
dcm = DObject.parse(objects[i], :no_meta => true, :syntax => transfer_syntaxes[i])
|
82
|
+
# Reset the logg threshold:
|
83
|
+
DICOM.logger.level = server_level
|
84
|
+
if dcm.read?
|
85
|
+
begin
|
86
|
+
message = self.save_file(path, dcm, transfer_syntaxes[i])
|
87
|
+
successful += 1
|
88
|
+
rescue
|
89
|
+
handle_fail += 1
|
90
|
+
all_success = false
|
91
|
+
messages << [:error, "Saving file failed!"]
|
92
|
+
end
|
93
|
+
else
|
94
|
+
parse_fail += 1
|
95
|
+
all_success = false
|
96
|
+
messages << [:error, "Invalid DICOM data encountered: The received string was not parsed successfully."]
|
97
|
+
end
|
98
|
+
else
|
99
|
+
too_short += 1
|
100
|
+
all_success = false
|
101
|
+
messages << [:error, "Invalid data encountered: The received string was too small to contain any DICOM data."]
|
102
|
+
end
|
103
|
+
end
|
104
|
+
# Create a summary status message, when multiple files have been received:
|
105
|
+
if total > 1
|
106
|
+
if successful == total
|
107
|
+
messages << [:info, "All #{total} DICOM files received successfully."]
|
108
|
+
else
|
109
|
+
if successful == 0
|
110
|
+
messages << [:warn, "All #{total} received DICOM files failed!"]
|
111
|
+
else
|
112
|
+
messages << [:warn, "Only #{successful} of #{total} DICOM files received successfully!"]
|
113
|
+
end
|
114
|
+
end
|
115
|
+
else
|
116
|
+
messages = [message] if all_success
|
117
|
+
end
|
118
|
+
return all_success, messages
|
119
|
+
end
|
120
|
+
|
121
|
+
end
|
122
122
|
end
|
@@ -1,210 +1,210 @@
|
|
1
|
-
module DICOM
|
2
|
-
|
3
|
-
# Ruby DICOM's registered DICOM UID root (Implementation Class UID).
|
4
|
-
UID_ROOT = "1.2.826.0.1.3680043.8.641"
|
5
|
-
# Ruby DICOM name & version (max 16 characters).
|
6
|
-
NAME = "RUBY-DCM_" + DICOM::VERSION
|
7
|
-
|
8
|
-
# Item tag.
|
9
|
-
ITEM_TAG = "FFFE,E000"
|
10
|
-
# All Item related tags (includes both types of delimitation items).
|
11
|
-
ITEM_TAGS = ["FFFE,E000", "FFFE,E00D", "FFFE,E0DD"]
|
12
|
-
# Item delimiter tag.
|
13
|
-
ITEM_DELIMITER = "FFFE,E00D"
|
14
|
-
# Sequence delimiter tag.
|
15
|
-
SEQUENCE_DELIMITER = "FFFE,E0DD"
|
16
|
-
# All delimiter tags.
|
17
|
-
DELIMITER_TAGS = ["FFFE,E00D", "FFFE,E0DD"]
|
18
|
-
|
19
|
-
# The VR used for the item elements.
|
20
|
-
ITEM_VR = " "
|
21
|
-
|
22
|
-
# Pixel tag.
|
23
|
-
PIXEL_TAG = "7FE0,0010"
|
24
|
-
# Name of the pixel tag when holding encapsulated data.
|
25
|
-
ENCAPSULATED_PIXEL_NAME = "Encapsulated Pixel Data"
|
26
|
-
# Name of encapsulated items.
|
27
|
-
PIXEL_ITEM_NAME = "Pixel Data Item"
|
28
|
-
|
29
|
-
# File meta group.
|
30
|
-
META_GROUP = "0002"
|
31
|
-
|
32
|
-
# Group length element.
|
33
|
-
GROUP_LENGTH = "0000"
|
34
|
-
|
35
|
-
# Implicit, little endian (the default transfer syntax).
|
36
|
-
IMPLICIT_LITTLE_ENDIAN = "1.2.840.10008.1.2"
|
37
|
-
# Explicit, little endian transfer syntax.
|
38
|
-
EXPLICIT_LITTLE_ENDIAN = "1.2.840.10008.1.2.1"
|
39
|
-
# Explicit, big endian transfer syntax.
|
40
|
-
EXPLICIT_BIG_ENDIAN = "1.2.840.10008.1.2.2"
|
41
|
-
|
42
|
-
# Verification SOP class UID.
|
43
|
-
VERIFICATION_SOP = "1.2.840.10008.1.1"
|
44
|
-
# Application context SOP class UID.
|
45
|
-
APPLICATION_CONTEXT = "1.2.840.10008.3.1.1.1"
|
46
|
-
|
47
|
-
# Network transmission successful.
|
48
|
-
SUCCESS = 0
|
49
|
-
# Network proposition accepted.
|
50
|
-
ACCEPTANCE = 0
|
51
|
-
# Presentation context rejected by abstract syntax.
|
52
|
-
ABSTRACT_SYNTAX_REJECTED = 3
|
53
|
-
# Presentation context rejected by transfer syntax.
|
54
|
-
TRANSFER_SYNTAX_REJECTED = 4
|
55
|
-
|
56
|
-
# Some network command element codes:
|
57
|
-
C_STORE_RQ = 1 # (encodes to 0001H as US)
|
58
|
-
C_GET_RQ = 16 # (encodes to 0010H as US)
|
59
|
-
C_FIND_RQ = 32 # (encodes to 0020H as US)
|
60
|
-
C_MOVE_RQ = 33 # (encodes to 0021H as US)
|
61
|
-
C_ECHO_RQ = 48 # (encodes to 0030 as US)
|
62
|
-
C_CANCEL_RQ = 4095 # (encodes to 0FFFH as US)
|
63
|
-
C_STORE_RSP = 32769 # (encodes to 8001H as US)
|
64
|
-
C_GET_RSP = 32784 # (encodes to 8010H as US)
|
65
|
-
C_FIND_RSP = 32800 # (encodes to 8020H as US)
|
66
|
-
C_MOVE_RSP = 32801 # (encodes to 8021H as US)
|
67
|
-
C_ECHO_RSP = 32816 # (encodes to 8030H as US)
|
68
|
-
NO_DATA_SET_PRESENT = 257 # (encodes to 0101H as US)
|
69
|
-
DATA_SET_PRESENT = 1
|
70
|
-
DEFAULT_MESSAGE_ID = 1
|
71
|
-
|
72
|
-
# The network communication flags:
|
73
|
-
DATA_MORE_FRAGMENTS = "00"
|
74
|
-
COMMAND_MORE_FRAGMENTS = "01"
|
75
|
-
DATA_LAST_FRAGMENT = "02"
|
76
|
-
COMMAND_LAST_FRAGMENT = "03"
|
77
|
-
|
78
|
-
# Network communication PDU types:
|
79
|
-
PDU_ASSOCIATION_REQUEST = "01"
|
80
|
-
PDU_ASSOCIATION_ACCEPT = "02"
|
81
|
-
PDU_ASSOCIATION_REJECT = "03"
|
82
|
-
PDU_DATA = "04"
|
83
|
-
PDU_RELEASE_REQUEST = "05"
|
84
|
-
PDU_RELEASE_RESPONSE = "06"
|
85
|
-
PDU_ABORT = "07"
|
86
|
-
|
87
|
-
# Network communication item types:
|
88
|
-
ITEM_APPLICATION_CONTEXT = "10"
|
89
|
-
ITEM_PRESENTATION_CONTEXT_REQUEST = "20"
|
90
|
-
ITEM_PRESENTATION_CONTEXT_RESPONSE = "21"
|
91
|
-
ITEM_ABSTRACT_SYNTAX = "30"
|
92
|
-
ITEM_TRANSFER_SYNTAX = "40"
|
93
|
-
ITEM_USER_INFORMATION = "50"
|
94
|
-
ITEM_MAX_LENGTH = "51"
|
95
|
-
ITEM_IMPLEMENTATION_UID = "52"
|
96
|
-
ITEM_MAX_OPERATIONS_INVOKED = "53"
|
97
|
-
ITEM_ROLE_NEGOTIATION = "54"
|
98
|
-
ITEM_IMPLEMENTATION_VERSION = "55"
|
99
|
-
|
100
|
-
# Varaibles used to determine endianness.
|
101
|
-
x = 0xdeadbeef
|
102
|
-
endian_type = {
|
103
|
-
Array(x).pack("V*") => false, # Little
|
104
|
-
Array(x).pack("N*") => true # Big
|
105
|
-
}
|
106
|
-
# System (CPU) Endianness.
|
107
|
-
CPU_ENDIAN = endian_type[Array(x).pack("L*")]
|
108
|
-
|
109
|
-
# Transfer Syntaxes (taken from the DICOM Specification PS 3.5, Chapter 10).
|
110
|
-
|
111
|
-
# General
|
112
|
-
TXS_IMPLICIT_LITTLE_ENDIAN = '1.2.840.10008.1.2' # also defined as IMPLICIT_LITTLE_ENDIAN, default transfer syntax
|
113
|
-
TXS_EXPLICIT_LITTLE_ENDIAN = '1.2.840.10008.1.2.1' # also defined as EXPLICIT_LITTLE_ENDIAN
|
114
|
-
TXS_EXPLICIT_BIG_ENDIAN = '1.2.840.10008.1.2.2' # also defined as EXPLICIT_BIG_ENDIAN
|
115
|
-
|
116
|
-
# TRANSFER SYNTAXES FOR ENCAPSULATION OF ENCODED PIXEL DATA
|
117
|
-
TXS_JPEG_BASELINE = '1.2.840.10008.1.2.4.50'
|
118
|
-
TXS_JPEG_EXTENDED = '1.2.840.10008.1.2.4.51'
|
119
|
-
TXS_JPEG_LOSSLESS_NH = '1.2.840.10008.1.2.4.57' # NH: non-hirarchical
|
120
|
-
TXS_JPEG_LOSSLESS_NH_FOP = '1.2.840.10008.1.2.4.70' # NH: non-hirarchical, FOP: first-order prediction
|
121
|
-
|
122
|
-
TXS_JPEG_LS_LOSSLESS = '1.2.840.10008.1.2.4.80'
|
123
|
-
TXS_JPEG_LS_NEAR_LOSSLESS = '1.2.840.10008.1.2.4.81'
|
124
|
-
|
125
|
-
TXS_JPEG_2000_PART1_LOSSLESS = '1.2.840.10008.1.2.4.90'
|
126
|
-
TXS_JPEG_2000_PART1_LOSSLESS_OR_LOSSY = '1.2.840.10008.1.2.4.91'
|
127
|
-
TXS_JPEG_2000_PART2_LOSSLESS = '1.2.840.10008.1.2.4.92'
|
128
|
-
TXS_JPEG_2000_PART2_LOSSLESS_OR_LOSSY = '1.2.840.10008.1.2.4.93'
|
129
|
-
|
130
|
-
TXS_MPEG2_MP_ML = '1.2.840.10008.1.2.4.100'
|
131
|
-
TXS_MPEG2_MP_HL = '1.2.840.10008.1.2.4.101'
|
132
|
-
|
133
|
-
TXS_DEFLATED_LITTLE_ENDIAN = '1.2.840.10008.1.2.1.99' # ZIP Compression
|
134
|
-
|
135
|
-
TXS_JPIP = '1.2.840.10008.1.2.4.94'
|
136
|
-
TXS_JPIP_DEFLATE = '1.2.840.10008.1.2.4.95'
|
137
|
-
|
138
|
-
TXS_RLE = '1.2.840.10008.1.2.5'
|
139
|
-
|
140
|
-
|
141
|
-
# Photometric Interpretations
|
142
|
-
# Taken from DICOM Specification PS 3.3 C.7.6.3.1.2 Photometric Interpretation
|
143
|
-
|
144
|
-
PI_MONOCHROME1 = 'MONOCHROME1'
|
145
|
-
PI_MONOCHROME2 = 'MONOCHROME2'
|
146
|
-
PI_PALETTE_COLOR = 'PALETTE COLOR'
|
147
|
-
PI_RGB = 'RGB'
|
148
|
-
PI_YBR_FULL = 'YBR_FULL'
|
149
|
-
PI_YBR_FULL_422 = 'YBR_FULL_422 '
|
150
|
-
PI_YBR_PARTIAL_422 = 'YBR_PARTIAL_422'
|
151
|
-
PI_YBR_PARTIAL_420 = 'YBR_PARTIAL_420'
|
152
|
-
PI_YBR_ICT = 'YBR_ICT'
|
153
|
-
PI_YBR_RCT = 'YBR_RCT'
|
154
|
-
|
155
|
-
# Retired Photometric Interpretations, are those needed to be supported?
|
156
|
-
PI_HSV = 'HSV'
|
157
|
-
PI_ARGB = 'ARGB'
|
158
|
-
PI_CMYK = 'CMYK'
|
159
|
-
|
160
|
-
# The relationship between DICOM Character Set and Encoding name.
|
161
|
-
ENCODING_NAME = {
|
162
|
-
'ISO_IR 100' => 'ISO-8859-1',
|
163
|
-
'ISO_IR 101' => 'ISO-8859-2',
|
164
|
-
'ISO_IR 109' => 'ISO-8859-3',
|
165
|
-
'ISO_IR 110' => 'ISO-8859-4',
|
166
|
-
'ISO_IR 144' => 'ISO-8859-5',
|
167
|
-
'ISO_IR 127' => 'ISO-8859-6',
|
168
|
-
'ISO_IR 126' => 'ISO-8859-7',
|
169
|
-
'ISO_IR 138' => 'ISO-8859-8',
|
170
|
-
'ISO_IR 148' => 'ISO-8859-9',
|
171
|
-
'ISO_IR 13' => 'JIS_X0201',
|
172
|
-
'ISO_IR 166' => 'ISO-8859-11',
|
173
|
-
'GB18030' => 'GB18030',
|
174
|
-
'ISO_IR 192' => 'UTF-8'
|
175
|
-
}
|
176
|
-
ENCODING_NAME.default = 'ASCII-8BIT'
|
177
|
-
|
178
|
-
# The type conversion (method) used for the various value representations.
|
179
|
-
VALUE_CONVERSION = {
|
180
|
-
'BY' => :to_i,
|
181
|
-
'US' => :to_i,
|
182
|
-
'SS' => :to_i,
|
183
|
-
'UL' => :to_i,
|
184
|
-
'SL' => :to_i,
|
185
|
-
'OB' => :to_i,
|
186
|
-
'OW' => :to_i,
|
187
|
-
'OF' => :to_f,
|
188
|
-
'FL' => :to_f,
|
189
|
-
'FD' => :to_f,
|
190
|
-
'AT' => :to_s,
|
191
|
-
'AE' => :to_s,
|
192
|
-
'AS' => :to_s,
|
193
|
-
'CS' => :to_s,
|
194
|
-
'DA' => :to_s,
|
195
|
-
'DS' => :to_s,
|
196
|
-
'DT' => :to_s,
|
197
|
-
'IS' => :to_s,
|
198
|
-
'LO' => :to_s,
|
199
|
-
'LT' => :to_s,
|
200
|
-
'PN' => :to_s,
|
201
|
-
'SH' => :to_s,
|
202
|
-
'ST' => :to_s,
|
203
|
-
'TM' => :to_s,
|
204
|
-
'UI' => :to_s,
|
205
|
-
'UT' => :to_s
|
206
|
-
}
|
207
|
-
VALUE_CONVERSION.default = :to_s
|
208
|
-
|
209
|
-
end
|
210
|
-
|
1
|
+
module DICOM
|
2
|
+
|
3
|
+
# Ruby DICOM's registered DICOM UID root (Implementation Class UID).
|
4
|
+
UID_ROOT = "1.2.826.0.1.3680043.8.641"
|
5
|
+
# Ruby DICOM name & version (max 16 characters).
|
6
|
+
NAME = "RUBY-DCM_" + DICOM::VERSION
|
7
|
+
|
8
|
+
# Item tag.
|
9
|
+
ITEM_TAG = "FFFE,E000"
|
10
|
+
# All Item related tags (includes both types of delimitation items).
|
11
|
+
ITEM_TAGS = ["FFFE,E000", "FFFE,E00D", "FFFE,E0DD"]
|
12
|
+
# Item delimiter tag.
|
13
|
+
ITEM_DELIMITER = "FFFE,E00D"
|
14
|
+
# Sequence delimiter tag.
|
15
|
+
SEQUENCE_DELIMITER = "FFFE,E0DD"
|
16
|
+
# All delimiter tags.
|
17
|
+
DELIMITER_TAGS = ["FFFE,E00D", "FFFE,E0DD"]
|
18
|
+
|
19
|
+
# The VR used for the item elements.
|
20
|
+
ITEM_VR = " "
|
21
|
+
|
22
|
+
# Pixel tag.
|
23
|
+
PIXEL_TAG = "7FE0,0010"
|
24
|
+
# Name of the pixel tag when holding encapsulated data.
|
25
|
+
ENCAPSULATED_PIXEL_NAME = "Encapsulated Pixel Data"
|
26
|
+
# Name of encapsulated items.
|
27
|
+
PIXEL_ITEM_NAME = "Pixel Data Item"
|
28
|
+
|
29
|
+
# File meta group.
|
30
|
+
META_GROUP = "0002"
|
31
|
+
|
32
|
+
# Group length element.
|
33
|
+
GROUP_LENGTH = "0000"
|
34
|
+
|
35
|
+
# Implicit, little endian (the default transfer syntax).
|
36
|
+
IMPLICIT_LITTLE_ENDIAN = "1.2.840.10008.1.2"
|
37
|
+
# Explicit, little endian transfer syntax.
|
38
|
+
EXPLICIT_LITTLE_ENDIAN = "1.2.840.10008.1.2.1"
|
39
|
+
# Explicit, big endian transfer syntax.
|
40
|
+
EXPLICIT_BIG_ENDIAN = "1.2.840.10008.1.2.2"
|
41
|
+
|
42
|
+
# Verification SOP class UID.
|
43
|
+
VERIFICATION_SOP = "1.2.840.10008.1.1"
|
44
|
+
# Application context SOP class UID.
|
45
|
+
APPLICATION_CONTEXT = "1.2.840.10008.3.1.1.1"
|
46
|
+
|
47
|
+
# Network transmission successful.
|
48
|
+
SUCCESS = 0
|
49
|
+
# Network proposition accepted.
|
50
|
+
ACCEPTANCE = 0
|
51
|
+
# Presentation context rejected by abstract syntax.
|
52
|
+
ABSTRACT_SYNTAX_REJECTED = 3
|
53
|
+
# Presentation context rejected by transfer syntax.
|
54
|
+
TRANSFER_SYNTAX_REJECTED = 4
|
55
|
+
|
56
|
+
# Some network command element codes:
|
57
|
+
C_STORE_RQ = 1 # (encodes to 0001H as US)
|
58
|
+
C_GET_RQ = 16 # (encodes to 0010H as US)
|
59
|
+
C_FIND_RQ = 32 # (encodes to 0020H as US)
|
60
|
+
C_MOVE_RQ = 33 # (encodes to 0021H as US)
|
61
|
+
C_ECHO_RQ = 48 # (encodes to 0030 as US)
|
62
|
+
C_CANCEL_RQ = 4095 # (encodes to 0FFFH as US)
|
63
|
+
C_STORE_RSP = 32769 # (encodes to 8001H as US)
|
64
|
+
C_GET_RSP = 32784 # (encodes to 8010H as US)
|
65
|
+
C_FIND_RSP = 32800 # (encodes to 8020H as US)
|
66
|
+
C_MOVE_RSP = 32801 # (encodes to 8021H as US)
|
67
|
+
C_ECHO_RSP = 32816 # (encodes to 8030H as US)
|
68
|
+
NO_DATA_SET_PRESENT = 257 # (encodes to 0101H as US)
|
69
|
+
DATA_SET_PRESENT = 1
|
70
|
+
DEFAULT_MESSAGE_ID = 1
|
71
|
+
|
72
|
+
# The network communication flags:
|
73
|
+
DATA_MORE_FRAGMENTS = "00"
|
74
|
+
COMMAND_MORE_FRAGMENTS = "01"
|
75
|
+
DATA_LAST_FRAGMENT = "02"
|
76
|
+
COMMAND_LAST_FRAGMENT = "03"
|
77
|
+
|
78
|
+
# Network communication PDU types:
|
79
|
+
PDU_ASSOCIATION_REQUEST = "01"
|
80
|
+
PDU_ASSOCIATION_ACCEPT = "02"
|
81
|
+
PDU_ASSOCIATION_REJECT = "03"
|
82
|
+
PDU_DATA = "04"
|
83
|
+
PDU_RELEASE_REQUEST = "05"
|
84
|
+
PDU_RELEASE_RESPONSE = "06"
|
85
|
+
PDU_ABORT = "07"
|
86
|
+
|
87
|
+
# Network communication item types:
|
88
|
+
ITEM_APPLICATION_CONTEXT = "10"
|
89
|
+
ITEM_PRESENTATION_CONTEXT_REQUEST = "20"
|
90
|
+
ITEM_PRESENTATION_CONTEXT_RESPONSE = "21"
|
91
|
+
ITEM_ABSTRACT_SYNTAX = "30"
|
92
|
+
ITEM_TRANSFER_SYNTAX = "40"
|
93
|
+
ITEM_USER_INFORMATION = "50"
|
94
|
+
ITEM_MAX_LENGTH = "51"
|
95
|
+
ITEM_IMPLEMENTATION_UID = "52"
|
96
|
+
ITEM_MAX_OPERATIONS_INVOKED = "53"
|
97
|
+
ITEM_ROLE_NEGOTIATION = "54"
|
98
|
+
ITEM_IMPLEMENTATION_VERSION = "55"
|
99
|
+
|
100
|
+
# Varaibles used to determine endianness.
|
101
|
+
x = 0xdeadbeef
|
102
|
+
endian_type = {
|
103
|
+
Array(x).pack("V*") => false, # Little
|
104
|
+
Array(x).pack("N*") => true # Big
|
105
|
+
}
|
106
|
+
# System (CPU) Endianness.
|
107
|
+
CPU_ENDIAN = endian_type[Array(x).pack("L*")]
|
108
|
+
|
109
|
+
# Transfer Syntaxes (taken from the DICOM Specification PS 3.5, Chapter 10).
|
110
|
+
|
111
|
+
# General
|
112
|
+
TXS_IMPLICIT_LITTLE_ENDIAN = '1.2.840.10008.1.2' # also defined as IMPLICIT_LITTLE_ENDIAN, default transfer syntax
|
113
|
+
TXS_EXPLICIT_LITTLE_ENDIAN = '1.2.840.10008.1.2.1' # also defined as EXPLICIT_LITTLE_ENDIAN
|
114
|
+
TXS_EXPLICIT_BIG_ENDIAN = '1.2.840.10008.1.2.2' # also defined as EXPLICIT_BIG_ENDIAN
|
115
|
+
|
116
|
+
# TRANSFER SYNTAXES FOR ENCAPSULATION OF ENCODED PIXEL DATA
|
117
|
+
TXS_JPEG_BASELINE = '1.2.840.10008.1.2.4.50'
|
118
|
+
TXS_JPEG_EXTENDED = '1.2.840.10008.1.2.4.51'
|
119
|
+
TXS_JPEG_LOSSLESS_NH = '1.2.840.10008.1.2.4.57' # NH: non-hirarchical
|
120
|
+
TXS_JPEG_LOSSLESS_NH_FOP = '1.2.840.10008.1.2.4.70' # NH: non-hirarchical, FOP: first-order prediction
|
121
|
+
|
122
|
+
TXS_JPEG_LS_LOSSLESS = '1.2.840.10008.1.2.4.80'
|
123
|
+
TXS_JPEG_LS_NEAR_LOSSLESS = '1.2.840.10008.1.2.4.81'
|
124
|
+
|
125
|
+
TXS_JPEG_2000_PART1_LOSSLESS = '1.2.840.10008.1.2.4.90'
|
126
|
+
TXS_JPEG_2000_PART1_LOSSLESS_OR_LOSSY = '1.2.840.10008.1.2.4.91'
|
127
|
+
TXS_JPEG_2000_PART2_LOSSLESS = '1.2.840.10008.1.2.4.92'
|
128
|
+
TXS_JPEG_2000_PART2_LOSSLESS_OR_LOSSY = '1.2.840.10008.1.2.4.93'
|
129
|
+
|
130
|
+
TXS_MPEG2_MP_ML = '1.2.840.10008.1.2.4.100'
|
131
|
+
TXS_MPEG2_MP_HL = '1.2.840.10008.1.2.4.101'
|
132
|
+
|
133
|
+
TXS_DEFLATED_LITTLE_ENDIAN = '1.2.840.10008.1.2.1.99' # ZIP Compression
|
134
|
+
|
135
|
+
TXS_JPIP = '1.2.840.10008.1.2.4.94'
|
136
|
+
TXS_JPIP_DEFLATE = '1.2.840.10008.1.2.4.95'
|
137
|
+
|
138
|
+
TXS_RLE = '1.2.840.10008.1.2.5'
|
139
|
+
|
140
|
+
|
141
|
+
# Photometric Interpretations
|
142
|
+
# Taken from DICOM Specification PS 3.3 C.7.6.3.1.2 Photometric Interpretation
|
143
|
+
|
144
|
+
PI_MONOCHROME1 = 'MONOCHROME1'
|
145
|
+
PI_MONOCHROME2 = 'MONOCHROME2'
|
146
|
+
PI_PALETTE_COLOR = 'PALETTE COLOR'
|
147
|
+
PI_RGB = 'RGB'
|
148
|
+
PI_YBR_FULL = 'YBR_FULL'
|
149
|
+
PI_YBR_FULL_422 = 'YBR_FULL_422 '
|
150
|
+
PI_YBR_PARTIAL_422 = 'YBR_PARTIAL_422'
|
151
|
+
PI_YBR_PARTIAL_420 = 'YBR_PARTIAL_420'
|
152
|
+
PI_YBR_ICT = 'YBR_ICT'
|
153
|
+
PI_YBR_RCT = 'YBR_RCT'
|
154
|
+
|
155
|
+
# Retired Photometric Interpretations, are those needed to be supported?
|
156
|
+
PI_HSV = 'HSV'
|
157
|
+
PI_ARGB = 'ARGB'
|
158
|
+
PI_CMYK = 'CMYK'
|
159
|
+
|
160
|
+
# The relationship between DICOM Character Set and Encoding name.
|
161
|
+
ENCODING_NAME = {
|
162
|
+
'ISO_IR 100' => 'ISO-8859-1',
|
163
|
+
'ISO_IR 101' => 'ISO-8859-2',
|
164
|
+
'ISO_IR 109' => 'ISO-8859-3',
|
165
|
+
'ISO_IR 110' => 'ISO-8859-4',
|
166
|
+
'ISO_IR 144' => 'ISO-8859-5',
|
167
|
+
'ISO_IR 127' => 'ISO-8859-6',
|
168
|
+
'ISO_IR 126' => 'ISO-8859-7',
|
169
|
+
'ISO_IR 138' => 'ISO-8859-8',
|
170
|
+
'ISO_IR 148' => 'ISO-8859-9',
|
171
|
+
'ISO_IR 13' => 'JIS_X0201',
|
172
|
+
'ISO_IR 166' => 'ISO-8859-11',
|
173
|
+
'GB18030' => 'GB18030',
|
174
|
+
'ISO_IR 192' => 'UTF-8'
|
175
|
+
}
|
176
|
+
ENCODING_NAME.default = 'ASCII-8BIT'
|
177
|
+
|
178
|
+
# The type conversion (method) used for the various value representations.
|
179
|
+
VALUE_CONVERSION = {
|
180
|
+
'BY' => :to_i,
|
181
|
+
'US' => :to_i,
|
182
|
+
'SS' => :to_i,
|
183
|
+
'UL' => :to_i,
|
184
|
+
'SL' => :to_i,
|
185
|
+
'OB' => :to_i,
|
186
|
+
'OW' => :to_i,
|
187
|
+
'OF' => :to_f,
|
188
|
+
'FL' => :to_f,
|
189
|
+
'FD' => :to_f,
|
190
|
+
'AT' => :to_s,
|
191
|
+
'AE' => :to_s,
|
192
|
+
'AS' => :to_s,
|
193
|
+
'CS' => :to_s,
|
194
|
+
'DA' => :to_s,
|
195
|
+
'DS' => :to_s,
|
196
|
+
'DT' => :to_s,
|
197
|
+
'IS' => :to_s,
|
198
|
+
'LO' => :to_s,
|
199
|
+
'LT' => :to_s,
|
200
|
+
'PN' => :to_s,
|
201
|
+
'SH' => :to_s,
|
202
|
+
'ST' => :to_s,
|
203
|
+
'TM' => :to_s,
|
204
|
+
'UI' => :to_s,
|
205
|
+
'UT' => :to_s
|
206
|
+
}
|
207
|
+
VALUE_CONVERSION.default = :to_s
|
208
|
+
|
209
|
+
end
|
210
|
+
|