Mercurial > hg > orthanc
comparison OrthancFramework/UnitTestsSources/FromDcmtkTests.cpp @ 4055:9214e3a7b0a2 framework
moving FromDcmtkTests.cpp from OrthancServer to OrthancFramework
author | Sebastien Jodogne <s.jodogne@gmail.com> |
---|---|
date | Thu, 11 Jun 2020 12:52:09 +0200 |
parents | OrthancServer/UnitTestsSources/FromDcmtkTests.cpp@05b8fd21089c |
children | 0953b3dc3261 |
comparison
equal
deleted
inserted
replaced
4054:9c37896a4457 | 4055:9214e3a7b0a2 |
---|---|
1 /** | |
2 * Orthanc - A Lightweight, RESTful DICOM Store | |
3 * Copyright (C) 2012-2016 Sebastien Jodogne, Medical Physics | |
4 * Department, University Hospital of Liege, Belgium | |
5 * Copyright (C) 2017-2020 Osimis S.A., Belgium | |
6 * | |
7 * This program is free software: you can redistribute it and/or | |
8 * modify it under the terms of the GNU General Public License as | |
9 * published by the Free Software Foundation, either version 3 of the | |
10 * License, or (at your option) any later version. | |
11 * | |
12 * In addition, as a special exception, the copyright holders of this | |
13 * program give permission to link the code of its release with the | |
14 * OpenSSL project's "OpenSSL" library (or with modified versions of it | |
15 * that use the same license as the "OpenSSL" library), and distribute | |
16 * the linked executables. You must obey the GNU General Public License | |
17 * in all respects for all of the code used other than "OpenSSL". If you | |
18 * modify file(s) with this exception, you may extend this exception to | |
19 * your version of the file(s), but you are not obligated to do so. If | |
20 * you do not wish to do so, delete this exception statement from your | |
21 * version. If you delete this exception statement from all source files | |
22 * in the program, then also delete it here. | |
23 * | |
24 * This program is distributed in the hope that it will be useful, but | |
25 * WITHOUT ANY WARRANTY; without even the implied warranty of | |
26 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU | |
27 * General Public License for more details. | |
28 * | |
29 * You should have received a copy of the GNU General Public License | |
30 * along with this program. If not, see <http://www.gnu.org/licenses/>. | |
31 **/ | |
32 | |
33 | |
34 #if ORTHANC_UNIT_TESTS_LINK_FRAMEWORK == 1 | |
35 # include <OrthancFramework.h> | |
36 #endif | |
37 | |
38 #include "gtest/gtest.h" | |
39 | |
40 #include "../Sources/Compatibility.h" | |
41 #include "../Sources/DicomNetworking/DicomFindAnswers.h" | |
42 #include "../Sources/DicomParsing/DicomModification.h" | |
43 #include "../Sources/DicomParsing/DicomWebJsonVisitor.h" | |
44 #include "../Sources/DicomParsing/FromDcmtkBridge.h" | |
45 #include "../Sources/DicomParsing/Internals/DicomImageDecoder.h" | |
46 #include "../Sources/DicomParsing/ToDcmtkBridge.h" | |
47 #include "../Sources/Endianness.h" | |
48 #include "../Sources/Images/Image.h" | |
49 #include "../Sources/Images/ImageBuffer.h" | |
50 #include "../Sources/Images/ImageProcessing.h" | |
51 #include "../Sources/Images/PngReader.h" | |
52 #include "../Sources/Images/PngWriter.h" | |
53 #include "../Sources/OrthancException.h" | |
54 #include "../Sources/SystemToolbox.h" | |
55 #include "../Resources/CodeGeneration/EncodingTests.h" | |
56 | |
57 #include <dcmtk/dcmdata/dcelem.h> | |
58 #include <dcmtk/dcmdata/dcdeftag.h> | |
59 #include <boost/algorithm/string/predicate.hpp> | |
60 | |
61 #if ORTHANC_ENABLE_PUGIXML == 1 | |
62 # include <pugixml.hpp> | |
63 #endif | |
64 | |
65 using namespace Orthanc; | |
66 | |
67 TEST(DicomFormat, Tag) | |
68 { | |
69 ASSERT_EQ("PatientName", FromDcmtkBridge::GetTagName(DicomTag(0x0010, 0x0010), "")); | |
70 | |
71 DicomTag t = FromDcmtkBridge::ParseTag("SeriesDescription"); | |
72 ASSERT_EQ(0x0008, t.GetGroup()); | |
73 ASSERT_EQ(0x103E, t.GetElement()); | |
74 | |
75 t = FromDcmtkBridge::ParseTag("0020-e040"); | |
76 ASSERT_EQ(0x0020, t.GetGroup()); | |
77 ASSERT_EQ(0xe040, t.GetElement()); | |
78 | |
79 // Test ==() and !=() operators | |
80 ASSERT_TRUE(DICOM_TAG_PATIENT_ID == DicomTag(0x0010, 0x0020)); | |
81 ASSERT_FALSE(DICOM_TAG_PATIENT_ID != DicomTag(0x0010, 0x0020)); | |
82 } | |
83 | |
84 | |
85 TEST(DicomModification, Basic) | |
86 { | |
87 DicomModification m; | |
88 m.SetupAnonymization(DicomVersion_2008); | |
89 //m.SetLevel(DicomRootLevel_Study); | |
90 //m.ReplacePlainString(DICOM_TAG_PATIENT_ID, "coucou"); | |
91 //m.ReplacePlainString(DICOM_TAG_PATIENT_NAME, "coucou"); | |
92 | |
93 ParsedDicomFile o(true); | |
94 o.SaveToFile("UnitTestsResults/anon.dcm"); | |
95 | |
96 for (int i = 0; i < 10; i++) | |
97 { | |
98 char b[1024]; | |
99 sprintf(b, "UnitTestsResults/anon%06d.dcm", i); | |
100 std::unique_ptr<ParsedDicomFile> f(o.Clone(false)); | |
101 if (i > 4) | |
102 o.ReplacePlainString(DICOM_TAG_SERIES_INSTANCE_UID, "coucou"); | |
103 m.Apply(*f); | |
104 f->SaveToFile(b); | |
105 } | |
106 } | |
107 | |
108 | |
109 TEST(DicomModification, Anonymization) | |
110 { | |
111 ASSERT_EQ(DICOM_TAG_PATIENT_NAME, FromDcmtkBridge::ParseTag("PatientName")); | |
112 | |
113 const DicomTag privateTag(0x0045, 0x1010); | |
114 const DicomTag privateTag2(FromDcmtkBridge::ParseTag("0031-1020")); | |
115 ASSERT_TRUE(privateTag.IsPrivate()); | |
116 ASSERT_TRUE(privateTag2.IsPrivate()); | |
117 ASSERT_EQ(0x0031, privateTag2.GetGroup()); | |
118 ASSERT_EQ(0x1020, privateTag2.GetElement()); | |
119 | |
120 std::string s; | |
121 ParsedDicomFile o(true); | |
122 o.ReplacePlainString(DICOM_TAG_PATIENT_NAME, "coucou"); | |
123 ASSERT_FALSE(o.GetTagValue(s, privateTag)); | |
124 o.Insert(privateTag, "private tag", false, "OrthancCreator"); | |
125 ASSERT_TRUE(o.GetTagValue(s, privateTag)); | |
126 ASSERT_STREQ("private tag", s.c_str()); | |
127 | |
128 ASSERT_FALSE(o.GetTagValue(s, privateTag2)); | |
129 ASSERT_THROW(o.Replace(privateTag2, std::string("hello"), false, DicomReplaceMode_ThrowIfAbsent, "OrthancCreator"), OrthancException); | |
130 ASSERT_FALSE(o.GetTagValue(s, privateTag2)); | |
131 o.Replace(privateTag2, std::string("hello"), false, DicomReplaceMode_IgnoreIfAbsent, "OrthancCreator"); | |
132 ASSERT_FALSE(o.GetTagValue(s, privateTag2)); | |
133 o.Replace(privateTag2, std::string("hello"), false, DicomReplaceMode_InsertIfAbsent, "OrthancCreator"); | |
134 ASSERT_TRUE(o.GetTagValue(s, privateTag2)); | |
135 ASSERT_STREQ("hello", s.c_str()); | |
136 o.Replace(privateTag2, std::string("hello world"), false, DicomReplaceMode_InsertIfAbsent, "OrthancCreator"); | |
137 ASSERT_TRUE(o.GetTagValue(s, privateTag2)); | |
138 ASSERT_STREQ("hello world", s.c_str()); | |
139 | |
140 ASSERT_TRUE(o.GetTagValue(s, DICOM_TAG_PATIENT_NAME)); | |
141 ASSERT_FALSE(Toolbox::IsUuid(s)); | |
142 | |
143 DicomModification m; | |
144 m.SetupAnonymization(DicomVersion_2008); | |
145 m.Keep(privateTag); | |
146 | |
147 m.Apply(o); | |
148 | |
149 ASSERT_TRUE(o.GetTagValue(s, DICOM_TAG_PATIENT_NAME)); | |
150 ASSERT_TRUE(Toolbox::IsUuid(s)); | |
151 ASSERT_TRUE(o.GetTagValue(s, privateTag)); | |
152 ASSERT_STREQ("private tag", s.c_str()); | |
153 | |
154 m.SetupAnonymization(DicomVersion_2008); | |
155 m.Apply(o); | |
156 ASSERT_FALSE(o.GetTagValue(s, privateTag)); | |
157 } | |
158 | |
159 | |
160 #include <dcmtk/dcmdata/dcuid.h> | |
161 | |
162 TEST(DicomModification, Png) | |
163 { | |
164 // Red dot in http://en.wikipedia.org/wiki/Data_URI_scheme (RGBA image) | |
165 std::string s = ""; | |
166 | |
167 std::string m, cc; | |
168 ASSERT_TRUE(Toolbox::DecodeDataUriScheme(m, cc, s)); | |
169 | |
170 ASSERT_EQ("image/png", m); | |
171 | |
172 PngReader reader; | |
173 reader.ReadFromMemory(cc); | |
174 | |
175 ASSERT_EQ(5u, reader.GetHeight()); | |
176 ASSERT_EQ(5u, reader.GetWidth()); | |
177 ASSERT_EQ(PixelFormat_RGBA32, reader.GetFormat()); | |
178 | |
179 ParsedDicomFile o(true); | |
180 o.EmbedContent(s); | |
181 o.SaveToFile("UnitTestsResults/png1.dcm"); | |
182 | |
183 // Red dot, without alpha channel | |
184 s = ""; | |
185 o.EmbedContent(s); | |
186 o.SaveToFile("UnitTestsResults/png2.dcm"); | |
187 | |
188 // Check box in Graylevel8 | |
189 s = ""; | |
190 o.EmbedContent(s); | |
191 //o.ReplacePlainString(DICOM_TAG_SOP_CLASS_UID, UID_DigitalXRayImageStorageForProcessing); | |
192 o.SaveToFile("UnitTestsResults/png3.dcm"); | |
193 | |
194 | |
195 { | |
196 // Gradient in Graylevel16 | |
197 | |
198 ImageBuffer img; | |
199 img.SetWidth(256); | |
200 img.SetHeight(256); | |
201 img.SetFormat(PixelFormat_Grayscale16); | |
202 | |
203 ImageAccessor accessor; | |
204 img.GetWriteableAccessor(accessor); | |
205 | |
206 uint16_t v = 0; | |
207 for (unsigned int y = 0; y < img.GetHeight(); y++) | |
208 { | |
209 uint16_t *p = reinterpret_cast<uint16_t*>(accessor.GetRow(y)); | |
210 for (unsigned int x = 0; x < img.GetWidth(); x++, p++, v++) | |
211 { | |
212 *p = v; | |
213 } | |
214 } | |
215 | |
216 o.EmbedImage(accessor); | |
217 o.SaveToFile("UnitTestsResults/png4.dcm"); | |
218 } | |
219 } | |
220 | |
221 | |
222 TEST(FromDcmtkBridge, Encodings1) | |
223 { | |
224 for (unsigned int i = 0; i < testEncodingsCount; i++) | |
225 { | |
226 std::string source(testEncodingsEncoded[i]); | |
227 std::string expected(testEncodingsExpected[i]); | |
228 std::string s = Toolbox::ConvertToUtf8(source, testEncodings[i], false); | |
229 //std::cout << EnumerationToString(testEncodings[i]) << std::endl; | |
230 EXPECT_EQ(expected, s); | |
231 } | |
232 } | |
233 | |
234 | |
235 TEST(FromDcmtkBridge, Enumerations) | |
236 { | |
237 // http://dicom.nema.org/medical/dicom/current/output/html/part03.html#sect_C.12.1.1.2 | |
238 Encoding e; | |
239 | |
240 ASSERT_FALSE(GetDicomEncoding(e, "")); | |
241 ASSERT_TRUE(GetDicomEncoding(e, "ISO_IR 6")); ASSERT_EQ(Encoding_Ascii, e); | |
242 | |
243 // http://dicom.nema.org/medical/dicom/current/output/html/part03.html#table_C.12-2 | |
244 ASSERT_TRUE(GetDicomEncoding(e, "ISO_IR 100")); ASSERT_EQ(Encoding_Latin1, e); | |
245 ASSERT_TRUE(GetDicomEncoding(e, "ISO_IR 101")); ASSERT_EQ(Encoding_Latin2, e); | |
246 ASSERT_TRUE(GetDicomEncoding(e, "ISO_IR 109")); ASSERT_EQ(Encoding_Latin3, e); | |
247 ASSERT_TRUE(GetDicomEncoding(e, "ISO_IR 110")); ASSERT_EQ(Encoding_Latin4, e); | |
248 ASSERT_TRUE(GetDicomEncoding(e, "ISO_IR 144")); ASSERT_EQ(Encoding_Cyrillic, e); | |
249 ASSERT_TRUE(GetDicomEncoding(e, "ISO_IR 127")); ASSERT_EQ(Encoding_Arabic, e); | |
250 ASSERT_TRUE(GetDicomEncoding(e, "ISO_IR 126")); ASSERT_EQ(Encoding_Greek, e); | |
251 ASSERT_TRUE(GetDicomEncoding(e, "ISO_IR 138")); ASSERT_EQ(Encoding_Hebrew, e); | |
252 ASSERT_TRUE(GetDicomEncoding(e, "ISO_IR 148")); ASSERT_EQ(Encoding_Latin5, e); | |
253 ASSERT_TRUE(GetDicomEncoding(e, "ISO_IR 13")); ASSERT_EQ(Encoding_Japanese, e); | |
254 ASSERT_TRUE(GetDicomEncoding(e, "ISO_IR 166")); ASSERT_EQ(Encoding_Thai, e); | |
255 | |
256 // http://dicom.nema.org/medical/dicom/current/output/html/part03.html#table_C.12-3 | |
257 ASSERT_TRUE(GetDicomEncoding(e, "ISO 2022 IR 6")); ASSERT_EQ(Encoding_Ascii, e); | |
258 ASSERT_TRUE(GetDicomEncoding(e, "ISO 2022 IR 100")); ASSERT_EQ(Encoding_Latin1, e); | |
259 ASSERT_TRUE(GetDicomEncoding(e, "ISO 2022 IR 101")); ASSERT_EQ(Encoding_Latin2, e); | |
260 ASSERT_TRUE(GetDicomEncoding(e, "ISO 2022 IR 109")); ASSERT_EQ(Encoding_Latin3, e); | |
261 ASSERT_TRUE(GetDicomEncoding(e, "ISO 2022 IR 110")); ASSERT_EQ(Encoding_Latin4, e); | |
262 ASSERT_TRUE(GetDicomEncoding(e, "ISO 2022 IR 144")); ASSERT_EQ(Encoding_Cyrillic, e); | |
263 ASSERT_TRUE(GetDicomEncoding(e, "ISO 2022 IR 127")); ASSERT_EQ(Encoding_Arabic, e); | |
264 ASSERT_TRUE(GetDicomEncoding(e, "ISO 2022 IR 126")); ASSERT_EQ(Encoding_Greek, e); | |
265 ASSERT_TRUE(GetDicomEncoding(e, "ISO 2022 IR 138")); ASSERT_EQ(Encoding_Hebrew, e); | |
266 ASSERT_TRUE(GetDicomEncoding(e, "ISO 2022 IR 148")); ASSERT_EQ(Encoding_Latin5, e); | |
267 ASSERT_TRUE(GetDicomEncoding(e, "ISO 2022 IR 13")); ASSERT_EQ(Encoding_Japanese, e); | |
268 ASSERT_TRUE(GetDicomEncoding(e, "ISO 2022 IR 166")); ASSERT_EQ(Encoding_Thai, e); | |
269 | |
270 // http://dicom.nema.org/medical/dicom/current/output/html/part03.html#table_C.12-4 | |
271 ASSERT_TRUE(GetDicomEncoding(e, "ISO 2022 IR 87")); ASSERT_EQ(Encoding_JapaneseKanji, e); | |
272 ASSERT_FALSE(GetDicomEncoding(e, "ISO 2022 IR 159")); //ASSERT_EQ(Encoding_JapaneseKanjiSupplementary, e); | |
273 ASSERT_TRUE(GetDicomEncoding(e, "ISO 2022 IR 149")); ASSERT_EQ(Encoding_Korean, e); | |
274 ASSERT_TRUE(GetDicomEncoding(e, "ISO 2022 IR 58")); ASSERT_EQ(Encoding_SimplifiedChinese, e); | |
275 | |
276 // http://dicom.nema.org/medical/dicom/current/output/html/part03.html#table_C.12-5 | |
277 ASSERT_TRUE(GetDicomEncoding(e, "ISO_IR 192")); ASSERT_EQ(Encoding_Utf8, e); | |
278 ASSERT_TRUE(GetDicomEncoding(e, "GB18030")); ASSERT_EQ(Encoding_Chinese, e); | |
279 ASSERT_TRUE(GetDicomEncoding(e, "GBK")); ASSERT_EQ(Encoding_Chinese, e); | |
280 } | |
281 | |
282 | |
283 TEST(FromDcmtkBridge, Encodings3) | |
284 { | |
285 for (unsigned int i = 0; i < testEncodingsCount; i++) | |
286 { | |
287 //std::cout << EnumerationToString(testEncodings[i]) << std::endl; | |
288 std::string dicom; | |
289 | |
290 { | |
291 ParsedDicomFile f(true); | |
292 f.SetEncoding(testEncodings[i]); | |
293 | |
294 std::string s = Toolbox::ConvertToUtf8(testEncodingsEncoded[i], testEncodings[i], false); | |
295 f.Insert(DICOM_TAG_PATIENT_NAME, s, false, ""); | |
296 f.SaveToMemoryBuffer(dicom); | |
297 } | |
298 | |
299 if (testEncodings[i] != Encoding_Windows1251) | |
300 { | |
301 ParsedDicomFile g(dicom); | |
302 | |
303 if (testEncodings[i] != Encoding_Ascii) | |
304 { | |
305 bool hasCodeExtensions; | |
306 ASSERT_EQ(testEncodings[i], g.DetectEncoding(hasCodeExtensions)); | |
307 ASSERT_FALSE(hasCodeExtensions); | |
308 } | |
309 | |
310 std::string tag; | |
311 ASSERT_TRUE(g.GetTagValue(tag, DICOM_TAG_PATIENT_NAME)); | |
312 ASSERT_EQ(std::string(testEncodingsExpected[i]), tag); | |
313 } | |
314 } | |
315 } | |
316 | |
317 | |
318 TEST(FromDcmtkBridge, ValueRepresentation) | |
319 { | |
320 ASSERT_EQ(ValueRepresentation_PersonName, | |
321 FromDcmtkBridge::LookupValueRepresentation(DICOM_TAG_PATIENT_NAME)); | |
322 ASSERT_EQ(ValueRepresentation_Date, | |
323 FromDcmtkBridge::LookupValueRepresentation(DicomTag(0x0008, 0x0020) /* StudyDate */)); | |
324 ASSERT_EQ(ValueRepresentation_Time, | |
325 FromDcmtkBridge::LookupValueRepresentation(DicomTag(0x0008, 0x0030) /* StudyTime */)); | |
326 ASSERT_EQ(ValueRepresentation_DateTime, | |
327 FromDcmtkBridge::LookupValueRepresentation(DicomTag(0x0008, 0x002a) /* AcquisitionDateTime */)); | |
328 ASSERT_EQ(ValueRepresentation_NotSupported, | |
329 FromDcmtkBridge::LookupValueRepresentation(DicomTag(0x0001, 0x0001) /* some private tag */)); | |
330 } | |
331 | |
332 | |
333 | |
334 static const DicomTag REFERENCED_STUDY_SEQUENCE(0x0008, 0x1110); | |
335 static const DicomTag REFERENCED_PATIENT_SEQUENCE(0x0008, 0x1120); | |
336 | |
337 static void CreateSampleJson(Json::Value& a) | |
338 { | |
339 { | |
340 Json::Value b = Json::objectValue; | |
341 b["PatientName"] = "Hello"; | |
342 b["PatientID"] = "World"; | |
343 b["StudyDescription"] = "Toto"; | |
344 a.append(b); | |
345 } | |
346 | |
347 { | |
348 Json::Value b = Json::objectValue; | |
349 b["PatientName"] = "data:application/octet-stream;base64,SGVsbG8y"; // echo -n "Hello2" | base64 | |
350 b["PatientID"] = "World2"; | |
351 a.append(b); | |
352 } | |
353 } | |
354 | |
355 | |
356 | |
357 TEST(ParsedDicomFile, InsertReplaceStrings) | |
358 { | |
359 ParsedDicomFile f(true); | |
360 | |
361 f.Insert(DICOM_TAG_PATIENT_NAME, "World", false, ""); | |
362 ASSERT_THROW(f.Insert(DICOM_TAG_PATIENT_ID, "Hello", false, ""), OrthancException); // Already existing tag | |
363 f.ReplacePlainString(DICOM_TAG_SOP_INSTANCE_UID, "Toto"); // (*) | |
364 f.ReplacePlainString(DICOM_TAG_SOP_CLASS_UID, "Tata"); // (**) | |
365 | |
366 std::string s; | |
367 ASSERT_TRUE(f.LookupTransferSyntax(s)); | |
368 // The default transfer syntax depends on the OS endianness | |
369 ASSERT_TRUE(s == GetTransferSyntaxUid(DicomTransferSyntax_LittleEndianExplicit) || | |
370 s == GetTransferSyntaxUid(DicomTransferSyntax_BigEndianExplicit)); | |
371 | |
372 ASSERT_THROW(f.Replace(DICOM_TAG_ACCESSION_NUMBER, std::string("Accession"), | |
373 false, DicomReplaceMode_ThrowIfAbsent, ""), OrthancException); | |
374 f.Replace(DICOM_TAG_ACCESSION_NUMBER, std::string("Accession"), false, DicomReplaceMode_IgnoreIfAbsent, ""); | |
375 ASSERT_FALSE(f.GetTagValue(s, DICOM_TAG_ACCESSION_NUMBER)); | |
376 f.Replace(DICOM_TAG_ACCESSION_NUMBER, std::string("Accession"), false, DicomReplaceMode_InsertIfAbsent, ""); | |
377 ASSERT_TRUE(f.GetTagValue(s, DICOM_TAG_ACCESSION_NUMBER)); | |
378 ASSERT_EQ(s, "Accession"); | |
379 f.Replace(DICOM_TAG_ACCESSION_NUMBER, std::string("Accession2"), false, DicomReplaceMode_IgnoreIfAbsent, ""); | |
380 ASSERT_TRUE(f.GetTagValue(s, DICOM_TAG_ACCESSION_NUMBER)); | |
381 ASSERT_EQ(s, "Accession2"); | |
382 f.Replace(DICOM_TAG_ACCESSION_NUMBER, std::string("Accession3"), false, DicomReplaceMode_ThrowIfAbsent, ""); | |
383 ASSERT_TRUE(f.GetTagValue(s, DICOM_TAG_ACCESSION_NUMBER)); | |
384 ASSERT_EQ(s, "Accession3"); | |
385 | |
386 ASSERT_TRUE(f.GetTagValue(s, DICOM_TAG_PATIENT_NAME)); | |
387 ASSERT_EQ(s, "World"); | |
388 ASSERT_TRUE(f.GetTagValue(s, DICOM_TAG_SOP_INSTANCE_UID)); | |
389 ASSERT_EQ(s, "Toto"); | |
390 ASSERT_TRUE(f.GetTagValue(s, DICOM_TAG_MEDIA_STORAGE_SOP_INSTANCE_UID)); // Implicitly modified by (*) | |
391 ASSERT_EQ(s, "Toto"); | |
392 ASSERT_TRUE(f.GetTagValue(s, DICOM_TAG_SOP_CLASS_UID)); | |
393 ASSERT_EQ(s, "Tata"); | |
394 ASSERT_TRUE(f.GetTagValue(s, DICOM_TAG_MEDIA_STORAGE_SOP_CLASS_UID)); // Implicitly modified by (**) | |
395 ASSERT_EQ(s, "Tata"); | |
396 } | |
397 | |
398 | |
399 | |
400 | |
401 TEST(ParsedDicomFile, InsertReplaceJson) | |
402 { | |
403 ParsedDicomFile f(true); | |
404 | |
405 Json::Value a; | |
406 CreateSampleJson(a); | |
407 | |
408 ASSERT_FALSE(f.HasTag(REFERENCED_STUDY_SEQUENCE)); | |
409 f.Remove(REFERENCED_STUDY_SEQUENCE); // No effect | |
410 f.Insert(REFERENCED_STUDY_SEQUENCE, a, true, ""); | |
411 ASSERT_TRUE(f.HasTag(REFERENCED_STUDY_SEQUENCE)); | |
412 ASSERT_THROW(f.Insert(REFERENCED_STUDY_SEQUENCE, a, true, ""), OrthancException); | |
413 f.Remove(REFERENCED_STUDY_SEQUENCE); | |
414 ASSERT_FALSE(f.HasTag(REFERENCED_STUDY_SEQUENCE)); | |
415 f.Insert(REFERENCED_STUDY_SEQUENCE, a, true, ""); | |
416 ASSERT_TRUE(f.HasTag(REFERENCED_STUDY_SEQUENCE)); | |
417 | |
418 ASSERT_FALSE(f.HasTag(REFERENCED_PATIENT_SEQUENCE)); | |
419 ASSERT_THROW(f.Replace(REFERENCED_PATIENT_SEQUENCE, a, false, DicomReplaceMode_ThrowIfAbsent, ""), OrthancException); | |
420 ASSERT_FALSE(f.HasTag(REFERENCED_PATIENT_SEQUENCE)); | |
421 f.Replace(REFERENCED_PATIENT_SEQUENCE, a, false, DicomReplaceMode_IgnoreIfAbsent, ""); | |
422 ASSERT_FALSE(f.HasTag(REFERENCED_PATIENT_SEQUENCE)); | |
423 f.Replace(REFERENCED_PATIENT_SEQUENCE, a, false, DicomReplaceMode_InsertIfAbsent, ""); | |
424 ASSERT_TRUE(f.HasTag(REFERENCED_PATIENT_SEQUENCE)); | |
425 | |
426 { | |
427 Json::Value b; | |
428 f.DatasetToJson(b, DicomToJsonFormat_Full, DicomToJsonFlags_Default, 0); | |
429 | |
430 Json::Value c; | |
431 Toolbox::SimplifyDicomAsJson(c, b, DicomToJsonFormat_Human); | |
432 | |
433 ASSERT_EQ(0, c["ReferencedPatientSequence"].compare(a)); | |
434 ASSERT_NE(0, c["ReferencedStudySequence"].compare(a)); // Because Data URI Scheme decoding was enabled | |
435 } | |
436 | |
437 a = "data:application/octet-stream;base64,VGF0YQ=="; // echo -n "Tata" | base64 | |
438 f.Replace(DICOM_TAG_SOP_INSTANCE_UID, a, false, DicomReplaceMode_InsertIfAbsent, ""); // (*) | |
439 f.Replace(DICOM_TAG_SOP_CLASS_UID, a, true, DicomReplaceMode_InsertIfAbsent, ""); // (**) | |
440 | |
441 std::string s; | |
442 ASSERT_TRUE(f.GetTagValue(s, DICOM_TAG_SOP_INSTANCE_UID)); | |
443 ASSERT_EQ(s, a.asString()); | |
444 ASSERT_TRUE(f.GetTagValue(s, DICOM_TAG_MEDIA_STORAGE_SOP_INSTANCE_UID)); // Implicitly modified by (*) | |
445 ASSERT_EQ(s, a.asString()); | |
446 ASSERT_TRUE(f.GetTagValue(s, DICOM_TAG_SOP_CLASS_UID)); | |
447 ASSERT_EQ(s, "Tata"); | |
448 ASSERT_TRUE(f.GetTagValue(s, DICOM_TAG_MEDIA_STORAGE_SOP_CLASS_UID)); // Implicitly modified by (**) | |
449 ASSERT_EQ(s, "Tata"); | |
450 } | |
451 | |
452 | |
453 TEST(ParsedDicomFile, JsonEncoding) | |
454 { | |
455 ParsedDicomFile f(true); | |
456 | |
457 for (unsigned int i = 0; i < testEncodingsCount; i++) | |
458 { | |
459 if (testEncodings[i] != Encoding_Windows1251) | |
460 { | |
461 //std::cout << EnumerationToString(testEncodings[i]) << std::endl; | |
462 f.SetEncoding(testEncodings[i]); | |
463 | |
464 if (testEncodings[i] != Encoding_Ascii) | |
465 { | |
466 bool hasCodeExtensions; | |
467 ASSERT_EQ(testEncodings[i], f.DetectEncoding(hasCodeExtensions)); | |
468 ASSERT_FALSE(hasCodeExtensions); | |
469 } | |
470 | |
471 Json::Value s = Toolbox::ConvertToUtf8(testEncodingsEncoded[i], testEncodings[i], false); | |
472 f.Replace(DICOM_TAG_PATIENT_NAME, s, false, DicomReplaceMode_InsertIfAbsent, ""); | |
473 | |
474 Json::Value v; | |
475 f.DatasetToJson(v, DicomToJsonFormat_Human, DicomToJsonFlags_Default, 0); | |
476 ASSERT_EQ(v["PatientName"].asString(), std::string(testEncodingsExpected[i])); | |
477 } | |
478 } | |
479 } | |
480 | |
481 | |
482 TEST(ParsedDicomFile, ToJsonFlags1) | |
483 { | |
484 FromDcmtkBridge::RegisterDictionaryTag(DicomTag(0x7053, 0x1000), ValueRepresentation_OtherByte, "MyPrivateTag", 1, 1, "OrthancCreator"); | |
485 FromDcmtkBridge::RegisterDictionaryTag(DicomTag(0x7050, 0x1000), ValueRepresentation_PersonName, "Declared public tag", 1, 1, ""); | |
486 | |
487 ParsedDicomFile f(true); | |
488 f.Insert(DicomTag(0x7050, 0x1000), "Some public tag", false, ""); // Even group => public tag | |
489 f.Insert(DicomTag(0x7052, 0x1000), "Some unknown tag", false, ""); // Even group => public, unknown tag | |
490 f.Insert(DicomTag(0x7053, 0x1000), "Some private tag", false, "OrthancCreator"); // Odd group => private tag | |
491 | |
492 Json::Value v; | |
493 f.DatasetToJson(v, DicomToJsonFormat_Short, DicomToJsonFlags_None, 0); | |
494 ASSERT_EQ(Json::objectValue, v.type()); | |
495 ASSERT_EQ(6u, v.getMemberNames().size()); | |
496 ASSERT_FALSE(v.isMember("7052,1000")); | |
497 ASSERT_FALSE(v.isMember("7053,1000")); | |
498 ASSERT_TRUE(v.isMember("7050,1000")); | |
499 ASSERT_EQ(Json::stringValue, v["7050,1000"].type()); | |
500 ASSERT_EQ("Some public tag", v["7050,1000"].asString()); | |
501 | |
502 f.DatasetToJson(v, DicomToJsonFormat_Short, static_cast<DicomToJsonFlags>(DicomToJsonFlags_IncludePrivateTags | DicomToJsonFlags_IncludeBinary | DicomToJsonFlags_ConvertBinaryToNull), 0); | |
503 ASSERT_EQ(Json::objectValue, v.type()); | |
504 ASSERT_EQ(7u, v.getMemberNames().size()); | |
505 ASSERT_FALSE(v.isMember("7052,1000")); | |
506 ASSERT_TRUE(v.isMember("7050,1000")); | |
507 ASSERT_TRUE(v.isMember("7053,1000")); | |
508 ASSERT_EQ("Some public tag", v["7050,1000"].asString()); | |
509 ASSERT_EQ(Json::nullValue, v["7053,1000"].type()); | |
510 | |
511 f.DatasetToJson(v, DicomToJsonFormat_Short, static_cast<DicomToJsonFlags>(DicomToJsonFlags_IncludePrivateTags), 0); | |
512 ASSERT_EQ(Json::objectValue, v.type()); | |
513 ASSERT_EQ(6u, v.getMemberNames().size()); | |
514 ASSERT_FALSE(v.isMember("7052,1000")); | |
515 ASSERT_TRUE(v.isMember("7050,1000")); | |
516 ASSERT_FALSE(v.isMember("7053,1000")); | |
517 | |
518 f.DatasetToJson(v, DicomToJsonFormat_Short, static_cast<DicomToJsonFlags>(DicomToJsonFlags_IncludePrivateTags | DicomToJsonFlags_IncludeBinary), 0); | |
519 ASSERT_EQ(Json::objectValue, v.type()); | |
520 ASSERT_EQ(7u, v.getMemberNames().size()); | |
521 ASSERT_FALSE(v.isMember("7052,1000")); | |
522 ASSERT_TRUE(v.isMember("7050,1000")); | |
523 ASSERT_TRUE(v.isMember("7053,1000")); | |
524 ASSERT_EQ("Some public tag", v["7050,1000"].asString()); | |
525 std::string mime, content; | |
526 ASSERT_EQ(Json::stringValue, v["7053,1000"].type()); | |
527 ASSERT_TRUE(Toolbox::DecodeDataUriScheme(mime, content, v["7053,1000"].asString())); | |
528 ASSERT_EQ("application/octet-stream", mime); | |
529 ASSERT_EQ("Some private tag", content); | |
530 | |
531 f.DatasetToJson(v, DicomToJsonFormat_Short, static_cast<DicomToJsonFlags>(DicomToJsonFlags_IncludeUnknownTags | DicomToJsonFlags_IncludeBinary | DicomToJsonFlags_ConvertBinaryToNull), 0); | |
532 ASSERT_EQ(Json::objectValue, v.type()); | |
533 ASSERT_EQ(7u, v.getMemberNames().size()); | |
534 ASSERT_TRUE(v.isMember("7050,1000")); | |
535 ASSERT_TRUE(v.isMember("7052,1000")); | |
536 ASSERT_FALSE(v.isMember("7053,1000")); | |
537 ASSERT_EQ("Some public tag", v["7050,1000"].asString()); | |
538 ASSERT_EQ(Json::nullValue, v["7052,1000"].type()); | |
539 | |
540 f.DatasetToJson(v, DicomToJsonFormat_Short, static_cast<DicomToJsonFlags>(DicomToJsonFlags_IncludeUnknownTags | DicomToJsonFlags_IncludeBinary), 0); | |
541 ASSERT_EQ(Json::objectValue, v.type()); | |
542 ASSERT_EQ(7u, v.getMemberNames().size()); | |
543 ASSERT_TRUE(v.isMember("7050,1000")); | |
544 ASSERT_TRUE(v.isMember("7052,1000")); | |
545 ASSERT_FALSE(v.isMember("7053,1000")); | |
546 ASSERT_EQ("Some public tag", v["7050,1000"].asString()); | |
547 ASSERT_EQ(Json::stringValue, v["7052,1000"].type()); | |
548 ASSERT_TRUE(Toolbox::DecodeDataUriScheme(mime, content, v["7052,1000"].asString())); | |
549 ASSERT_EQ("application/octet-stream", mime); | |
550 ASSERT_EQ("Some unknown tag", content); | |
551 | |
552 f.DatasetToJson(v, DicomToJsonFormat_Short, static_cast<DicomToJsonFlags>(DicomToJsonFlags_IncludeUnknownTags | DicomToJsonFlags_IncludePrivateTags | DicomToJsonFlags_IncludeBinary | DicomToJsonFlags_ConvertBinaryToNull), 0); | |
553 ASSERT_EQ(Json::objectValue, v.type()); | |
554 ASSERT_EQ(8u, v.getMemberNames().size()); | |
555 ASSERT_TRUE(v.isMember("7050,1000")); | |
556 ASSERT_TRUE(v.isMember("7052,1000")); | |
557 ASSERT_TRUE(v.isMember("7053,1000")); | |
558 ASSERT_EQ("Some public tag", v["7050,1000"].asString()); | |
559 ASSERT_EQ(Json::nullValue, v["7052,1000"].type()); | |
560 ASSERT_EQ(Json::nullValue, v["7053,1000"].type()); | |
561 } | |
562 | |
563 | |
564 TEST(ParsedDicomFile, ToJsonFlags2) | |
565 { | |
566 ParsedDicomFile f(true); | |
567 | |
568 { | |
569 // "ParsedDicomFile" uses Little Endian => 'B' (least significant | |
570 // byte) will be stored first in the memory buffer and in the | |
571 // file, then 'A'. Hence the expected "BA" value below. | |
572 Uint16 v[] = { 'A' * 256 + 'B', 0 }; | |
573 ASSERT_TRUE(f.GetDcmtkObject().getDataset()->putAndInsertUint16Array(DCM_PixelData, v, 2).good()); | |
574 } | |
575 | |
576 Json::Value v; | |
577 f.DatasetToJson(v, DicomToJsonFormat_Short, DicomToJsonFlags_None, 0); | |
578 ASSERT_EQ(Json::objectValue, v.type()); | |
579 ASSERT_EQ(5u, v.getMemberNames().size()); | |
580 ASSERT_FALSE(v.isMember("7fe0,0010")); | |
581 | |
582 f.DatasetToJson(v, DicomToJsonFormat_Short, static_cast<DicomToJsonFlags>(DicomToJsonFlags_IncludePixelData | DicomToJsonFlags_ConvertBinaryToNull), 0); | |
583 ASSERT_EQ(Json::objectValue, v.type()); | |
584 ASSERT_EQ(6u, v.getMemberNames().size()); | |
585 ASSERT_TRUE(v.isMember("7fe0,0010")); | |
586 ASSERT_EQ(Json::nullValue, v["7fe0,0010"].type()); | |
587 | |
588 f.DatasetToJson(v, DicomToJsonFormat_Short, static_cast<DicomToJsonFlags>(DicomToJsonFlags_IncludePixelData | DicomToJsonFlags_ConvertBinaryToAscii), 0); | |
589 ASSERT_EQ(Json::objectValue, v.type()); | |
590 ASSERT_EQ(6u, v.getMemberNames().size()); | |
591 ASSERT_TRUE(v.isMember("7fe0,0010")); | |
592 ASSERT_EQ(Json::stringValue, v["7fe0,0010"].type()); | |
593 ASSERT_EQ("BA", v["7fe0,0010"].asString().substr(0, 2)); | |
594 | |
595 f.DatasetToJson(v, DicomToJsonFormat_Short, DicomToJsonFlags_IncludePixelData, 0); | |
596 ASSERT_EQ(Json::objectValue, v.type()); | |
597 ASSERT_EQ(6u, v.getMemberNames().size()); | |
598 ASSERT_TRUE(v.isMember("7fe0,0010")); | |
599 ASSERT_EQ(Json::stringValue, v["7fe0,0010"].type()); | |
600 std::string mime, content; | |
601 ASSERT_TRUE(Toolbox::DecodeDataUriScheme(mime, content, v["7fe0,0010"].asString())); | |
602 ASSERT_EQ("application/octet-stream", mime); | |
603 ASSERT_EQ("BA", content.substr(0, 2)); | |
604 } | |
605 | |
606 | |
607 TEST(DicomFindAnswers, Basic) | |
608 { | |
609 DicomFindAnswers a(false); | |
610 | |
611 { | |
612 DicomMap m; | |
613 m.SetValue(DICOM_TAG_PATIENT_ID, "hello", false); | |
614 a.Add(m); | |
615 } | |
616 | |
617 { | |
618 ParsedDicomFile d(true); | |
619 d.ReplacePlainString(DICOM_TAG_PATIENT_ID, "my"); | |
620 a.Add(d); | |
621 } | |
622 | |
623 { | |
624 DicomMap m; | |
625 m.SetValue(DICOM_TAG_PATIENT_ID, "world", false); | |
626 a.Add(m); | |
627 } | |
628 | |
629 Json::Value j; | |
630 a.ToJson(j, true); | |
631 ASSERT_EQ(3u, j.size()); | |
632 | |
633 //std::cout << j; | |
634 } | |
635 | |
636 | |
637 TEST(ParsedDicomFile, FromJson) | |
638 { | |
639 FromDcmtkBridge::RegisterDictionaryTag(DicomTag(0x7057, 0x1000), ValueRepresentation_OtherByte, "MyPrivateTag2", 1, 1, "ORTHANC"); | |
640 FromDcmtkBridge::RegisterDictionaryTag(DicomTag(0x7059, 0x1000), ValueRepresentation_OtherByte, "MyPrivateTag3", 1, 1, ""); | |
641 FromDcmtkBridge::RegisterDictionaryTag(DicomTag(0x7050, 0x1000), ValueRepresentation_PersonName, "Declared public tag2", 1, 1, ""); | |
642 | |
643 Json::Value v; | |
644 const std::string sopClassUid = "1.2.840.10008.5.1.4.1.1.1"; // CR Image Storage: | |
645 | |
646 // Test the private creator | |
647 ASSERT_EQ(DcmTag_ERROR_TagName, FromDcmtkBridge::GetTagName(DicomTag(0x7057, 0x1000), "NOPE")); | |
648 ASSERT_EQ("MyPrivateTag2", FromDcmtkBridge::GetTagName(DicomTag(0x7057, 0x1000), "ORTHANC")); | |
649 | |
650 { | |
651 v["SOPClassUID"] = sopClassUid; | |
652 v["SpecificCharacterSet"] = "ISO_IR 148"; // This is latin-5 | |
653 v["PatientName"] = "Sébastien"; | |
654 v["7050-1000"] = "Some public tag"; // Even group => public tag | |
655 v["7052-1000"] = "Some unknown tag"; // Even group => public, unknown tag | |
656 v["7057-1000"] = "Some private tag"; // Odd group => private tag | |
657 v["7059-1000"] = "Some private tag2"; // Odd group => private tag, with an odd length to test padding | |
658 | |
659 std::string s; | |
660 Toolbox::EncodeDataUriScheme(s, "application/octet-stream", "Sebastien"); | |
661 v["StudyDescription"] = s; | |
662 | |
663 v["PixelData"] = ""; // A red dot of 5x5 pixels | |
664 v["0040,0100"] = Json::arrayValue; // ScheduledProcedureStepSequence | |
665 | |
666 Json::Value vv; | |
667 vv["Modality"] = "MR"; | |
668 v["0040,0100"].append(vv); | |
669 | |
670 vv["Modality"] = "CT"; | |
671 v["0040,0100"].append(vv); | |
672 } | |
673 | |
674 const DicomToJsonFlags toJsonFlags = static_cast<DicomToJsonFlags>(DicomToJsonFlags_IncludeBinary | | |
675 DicomToJsonFlags_IncludePixelData | | |
676 DicomToJsonFlags_IncludePrivateTags | | |
677 DicomToJsonFlags_IncludeUnknownTags | | |
678 DicomToJsonFlags_ConvertBinaryToAscii); | |
679 | |
680 | |
681 { | |
682 std::unique_ptr<ParsedDicomFile> dicom | |
683 (ParsedDicomFile::CreateFromJson(v, static_cast<DicomFromJsonFlags>(DicomFromJsonFlags_GenerateIdentifiers), "")); | |
684 | |
685 Json::Value vv; | |
686 dicom->DatasetToJson(vv, DicomToJsonFormat_Human, toJsonFlags, 0); | |
687 | |
688 ASSERT_EQ(vv["SOPClassUID"].asString(), sopClassUid); | |
689 ASSERT_EQ(vv["MediaStorageSOPClassUID"].asString(), sopClassUid); | |
690 ASSERT_TRUE(vv.isMember("SOPInstanceUID")); | |
691 ASSERT_TRUE(vv.isMember("SeriesInstanceUID")); | |
692 ASSERT_TRUE(vv.isMember("StudyInstanceUID")); | |
693 ASSERT_TRUE(vv.isMember("PatientID")); | |
694 } | |
695 | |
696 | |
697 { | |
698 std::unique_ptr<ParsedDicomFile> dicom | |
699 (ParsedDicomFile::CreateFromJson(v, static_cast<DicomFromJsonFlags>(DicomFromJsonFlags_GenerateIdentifiers), "")); | |
700 | |
701 Json::Value vv; | |
702 dicom->DatasetToJson(vv, DicomToJsonFormat_Human, static_cast<DicomToJsonFlags>(DicomToJsonFlags_IncludePixelData), 0); | |
703 | |
704 std::string mime, content; | |
705 ASSERT_TRUE(Toolbox::DecodeDataUriScheme(mime, content, vv["PixelData"].asString())); | |
706 ASSERT_EQ("application/octet-stream", mime); | |
707 ASSERT_EQ(5u * 5u * 3u /* the red dot is 5x5 pixels in RGB24 */ + 1 /* for padding */, content.size()); | |
708 } | |
709 | |
710 | |
711 { | |
712 std::unique_ptr<ParsedDicomFile> dicom | |
713 (ParsedDicomFile::CreateFromJson(v, static_cast<DicomFromJsonFlags>(DicomFromJsonFlags_DecodeDataUriScheme), "")); | |
714 | |
715 Json::Value vv; | |
716 dicom->DatasetToJson(vv, DicomToJsonFormat_Short, toJsonFlags, 0); | |
717 | |
718 ASSERT_FALSE(vv.isMember("SOPInstanceUID")); | |
719 ASSERT_FALSE(vv.isMember("SeriesInstanceUID")); | |
720 ASSERT_FALSE(vv.isMember("StudyInstanceUID")); | |
721 ASSERT_FALSE(vv.isMember("PatientID")); | |
722 ASSERT_EQ(2u, vv["0040,0100"].size()); | |
723 ASSERT_EQ("MR", vv["0040,0100"][0]["0008,0060"].asString()); | |
724 ASSERT_EQ("CT", vv["0040,0100"][1]["0008,0060"].asString()); | |
725 ASSERT_EQ("Some public tag", vv["7050,1000"].asString()); | |
726 ASSERT_EQ("Some unknown tag", vv["7052,1000"].asString()); | |
727 ASSERT_EQ("Some private tag", vv["7057,1000"].asString()); | |
728 ASSERT_EQ("Some private tag2", vv["7059,1000"].asString()); | |
729 ASSERT_EQ("Sébastien", vv["0010,0010"].asString()); | |
730 ASSERT_EQ("Sebastien", vv["0008,1030"].asString()); | |
731 ASSERT_EQ("ISO_IR 148", vv["0008,0005"].asString()); | |
732 ASSERT_EQ("5", vv[DICOM_TAG_ROWS.Format()].asString()); | |
733 ASSERT_EQ("5", vv[DICOM_TAG_COLUMNS.Format()].asString()); | |
734 ASSERT_TRUE(vv[DICOM_TAG_PIXEL_DATA.Format()].asString().empty()); | |
735 } | |
736 } | |
737 | |
738 | |
739 | |
740 TEST(TestImages, PatternGrayscale8) | |
741 { | |
742 static const char* PATH = "UnitTestsResults/PatternGrayscale8.dcm"; | |
743 | |
744 Orthanc::Image image(Orthanc::PixelFormat_Grayscale8, 256, 256, false); | |
745 | |
746 for (int y = 0; y < 256; y++) | |
747 { | |
748 uint8_t *p = reinterpret_cast<uint8_t*>(image.GetRow(y)); | |
749 for (int x = 0; x < 256; x++, p++) | |
750 { | |
751 *p = y; | |
752 } | |
753 } | |
754 | |
755 Orthanc::ImageAccessor r; | |
756 | |
757 image.GetRegion(r, 32, 32, 64, 192); | |
758 Orthanc::ImageProcessing::Set(r, 0); | |
759 | |
760 image.GetRegion(r, 160, 32, 64, 192); | |
761 Orthanc::ImageProcessing::Set(r, 255); | |
762 | |
763 { | |
764 ParsedDicomFile f(true); | |
765 f.ReplacePlainString(DICOM_TAG_SOP_CLASS_UID, "1.2.840.10008.5.1.4.1.1.7"); | |
766 f.ReplacePlainString(DICOM_TAG_STUDY_INSTANCE_UID, "1.2.276.0.7230010.3.1.2.2831176407.321.1458901422.884998"); | |
767 f.ReplacePlainString(DICOM_TAG_PATIENT_ID, "ORTHANC"); | |
768 f.ReplacePlainString(DICOM_TAG_PATIENT_NAME, "Orthanc"); | |
769 f.ReplacePlainString(DICOM_TAG_STUDY_DESCRIPTION, "Patterns"); | |
770 f.ReplacePlainString(DICOM_TAG_SERIES_DESCRIPTION, "Grayscale8"); | |
771 f.EmbedImage(image); | |
772 | |
773 f.SaveToFile(PATH); | |
774 } | |
775 | |
776 { | |
777 std::string s; | |
778 Orthanc::SystemToolbox::ReadFile(s, PATH); | |
779 Orthanc::ParsedDicomFile f(s); | |
780 | |
781 std::unique_ptr<Orthanc::ImageAccessor> decoded(Orthanc::DicomImageDecoder::Decode(f, 0)); | |
782 ASSERT_EQ(256u, decoded->GetWidth()); | |
783 ASSERT_EQ(256u, decoded->GetHeight()); | |
784 ASSERT_EQ(Orthanc::PixelFormat_Grayscale8, decoded->GetFormat()); | |
785 | |
786 for (int y = 0; y < 256; y++) | |
787 { | |
788 const void* a = image.GetConstRow(y); | |
789 const void* b = decoded->GetConstRow(y); | |
790 ASSERT_EQ(0, memcmp(a, b, 256)); | |
791 } | |
792 } | |
793 } | |
794 | |
795 | |
796 TEST(TestImages, PatternRGB) | |
797 { | |
798 static const char* PATH = "UnitTestsResults/PatternRGB24.dcm"; | |
799 | |
800 Orthanc::Image image(Orthanc::PixelFormat_RGB24, 384, 256, false); | |
801 | |
802 for (int y = 0; y < 256; y++) | |
803 { | |
804 uint8_t *p = reinterpret_cast<uint8_t*>(image.GetRow(y)); | |
805 for (int x = 0; x < 128; x++, p += 3) | |
806 { | |
807 p[0] = y; | |
808 p[1] = 0; | |
809 p[2] = 0; | |
810 } | |
811 for (int x = 128; x < 128 * 2; x++, p += 3) | |
812 { | |
813 p[0] = 0; | |
814 p[1] = 255 - y; | |
815 p[2] = 0; | |
816 } | |
817 for (int x = 128 * 2; x < 128 * 3; x++, p += 3) | |
818 { | |
819 p[0] = 0; | |
820 p[1] = 0; | |
821 p[2] = y; | |
822 } | |
823 } | |
824 | |
825 { | |
826 ParsedDicomFile f(true); | |
827 f.ReplacePlainString(DICOM_TAG_SOP_CLASS_UID, "1.2.840.10008.5.1.4.1.1.7"); | |
828 f.ReplacePlainString(DICOM_TAG_STUDY_INSTANCE_UID, "1.2.276.0.7230010.3.1.2.2831176407.321.1458901422.884998"); | |
829 f.ReplacePlainString(DICOM_TAG_PATIENT_ID, "ORTHANC"); | |
830 f.ReplacePlainString(DICOM_TAG_PATIENT_NAME, "Orthanc"); | |
831 f.ReplacePlainString(DICOM_TAG_STUDY_DESCRIPTION, "Patterns"); | |
832 f.ReplacePlainString(DICOM_TAG_SERIES_DESCRIPTION, "RGB24"); | |
833 f.EmbedImage(image); | |
834 | |
835 f.SaveToFile(PATH); | |
836 } | |
837 | |
838 { | |
839 std::string s; | |
840 Orthanc::SystemToolbox::ReadFile(s, PATH); | |
841 Orthanc::ParsedDicomFile f(s); | |
842 | |
843 std::unique_ptr<Orthanc::ImageAccessor> decoded(Orthanc::DicomImageDecoder::Decode(f, 0)); | |
844 ASSERT_EQ(384u, decoded->GetWidth()); | |
845 ASSERT_EQ(256u, decoded->GetHeight()); | |
846 ASSERT_EQ(Orthanc::PixelFormat_RGB24, decoded->GetFormat()); | |
847 | |
848 for (int y = 0; y < 256; y++) | |
849 { | |
850 const void* a = image.GetConstRow(y); | |
851 const void* b = decoded->GetConstRow(y); | |
852 ASSERT_EQ(0, memcmp(a, b, 3 * 384)); | |
853 } | |
854 } | |
855 } | |
856 | |
857 | |
858 TEST(TestImages, PatternUint16) | |
859 { | |
860 static const char* PATH = "UnitTestsResults/PatternGrayscale16.dcm"; | |
861 | |
862 Orthanc::Image image(Orthanc::PixelFormat_Grayscale16, 256, 256, false); | |
863 | |
864 uint16_t v = 0; | |
865 for (int y = 0; y < 256; y++) | |
866 { | |
867 uint16_t *p = reinterpret_cast<uint16_t*>(image.GetRow(y)); | |
868 for (int x = 0; x < 256; x++, v++, p++) | |
869 { | |
870 *p = htole16(v); // Orthanc uses Little-Endian transfer syntax to encode images | |
871 } | |
872 } | |
873 | |
874 Orthanc::ImageAccessor r; | |
875 | |
876 image.GetRegion(r, 32, 32, 64, 192); | |
877 Orthanc::ImageProcessing::Set(r, 0); | |
878 | |
879 image.GetRegion(r, 160, 32, 64, 192); | |
880 Orthanc::ImageProcessing::Set(r, 65535); | |
881 | |
882 { | |
883 ParsedDicomFile f(true); | |
884 f.ReplacePlainString(DICOM_TAG_SOP_CLASS_UID, "1.2.840.10008.5.1.4.1.1.7"); | |
885 f.ReplacePlainString(DICOM_TAG_STUDY_INSTANCE_UID, "1.2.276.0.7230010.3.1.2.2831176407.321.1458901422.884998"); | |
886 f.ReplacePlainString(DICOM_TAG_PATIENT_ID, "ORTHANC"); | |
887 f.ReplacePlainString(DICOM_TAG_PATIENT_NAME, "Orthanc"); | |
888 f.ReplacePlainString(DICOM_TAG_STUDY_DESCRIPTION, "Patterns"); | |
889 f.ReplacePlainString(DICOM_TAG_SERIES_DESCRIPTION, "Grayscale16"); | |
890 f.EmbedImage(image); | |
891 | |
892 f.SaveToFile(PATH); | |
893 } | |
894 | |
895 { | |
896 std::string s; | |
897 Orthanc::SystemToolbox::ReadFile(s, PATH); | |
898 Orthanc::ParsedDicomFile f(s); | |
899 | |
900 std::unique_ptr<Orthanc::ImageAccessor> decoded(Orthanc::DicomImageDecoder::Decode(f, 0)); | |
901 ASSERT_EQ(256u, decoded->GetWidth()); | |
902 ASSERT_EQ(256u, decoded->GetHeight()); | |
903 ASSERT_EQ(Orthanc::PixelFormat_Grayscale16, decoded->GetFormat()); | |
904 | |
905 for (int y = 0; y < 256; y++) | |
906 { | |
907 const void* a = image.GetConstRow(y); | |
908 const void* b = decoded->GetConstRow(y); | |
909 ASSERT_EQ(0, memcmp(a, b, 512)); | |
910 } | |
911 } | |
912 } | |
913 | |
914 | |
915 TEST(TestImages, PatternInt16) | |
916 { | |
917 static const char* PATH = "UnitTestsResults/PatternSignedGrayscale16.dcm"; | |
918 | |
919 Orthanc::Image image(Orthanc::PixelFormat_SignedGrayscale16, 256, 256, false); | |
920 | |
921 int16_t v = -32768; | |
922 for (int y = 0; y < 256; y++) | |
923 { | |
924 int16_t *p = reinterpret_cast<int16_t*>(image.GetRow(y)); | |
925 for (int x = 0; x < 256; x++, v++, p++) | |
926 { | |
927 *p = htole16(v); // Orthanc uses Little-Endian transfer syntax to encode images | |
928 } | |
929 } | |
930 | |
931 Orthanc::ImageAccessor r; | |
932 image.GetRegion(r, 32, 32, 64, 192); | |
933 Orthanc::ImageProcessing::Set(r, -32768); | |
934 | |
935 image.GetRegion(r, 160, 32, 64, 192); | |
936 Orthanc::ImageProcessing::Set(r, 32767); | |
937 | |
938 { | |
939 ParsedDicomFile f(true); | |
940 f.ReplacePlainString(DICOM_TAG_SOP_CLASS_UID, "1.2.840.10008.5.1.4.1.1.7"); | |
941 f.ReplacePlainString(DICOM_TAG_STUDY_INSTANCE_UID, "1.2.276.0.7230010.3.1.2.2831176407.321.1458901422.884998"); | |
942 f.ReplacePlainString(DICOM_TAG_PATIENT_ID, "ORTHANC"); | |
943 f.ReplacePlainString(DICOM_TAG_PATIENT_NAME, "Orthanc"); | |
944 f.ReplacePlainString(DICOM_TAG_STUDY_DESCRIPTION, "Patterns"); | |
945 f.ReplacePlainString(DICOM_TAG_SERIES_DESCRIPTION, "SignedGrayscale16"); | |
946 f.EmbedImage(image); | |
947 | |
948 f.SaveToFile(PATH); | |
949 } | |
950 | |
951 { | |
952 std::string s; | |
953 Orthanc::SystemToolbox::ReadFile(s, PATH); | |
954 Orthanc::ParsedDicomFile f(s); | |
955 | |
956 std::unique_ptr<Orthanc::ImageAccessor> decoded(Orthanc::DicomImageDecoder::Decode(f, 0)); | |
957 ASSERT_EQ(256u, decoded->GetWidth()); | |
958 ASSERT_EQ(256u, decoded->GetHeight()); | |
959 ASSERT_EQ(Orthanc::PixelFormat_SignedGrayscale16, decoded->GetFormat()); | |
960 | |
961 for (int y = 0; y < 256; y++) | |
962 { | |
963 const void* a = image.GetConstRow(y); | |
964 const void* b = decoded->GetConstRow(y); | |
965 ASSERT_EQ(0, memcmp(a, b, 512)); | |
966 } | |
967 } | |
968 } | |
969 | |
970 | |
971 | |
972 static void CheckEncoding(const ParsedDicomFile& dicom, | |
973 Encoding expected) | |
974 { | |
975 const char* value = NULL; | |
976 ASSERT_TRUE(dicom.GetDcmtkObject().getDataset()->findAndGetString(DCM_SpecificCharacterSet, value).good()); | |
977 | |
978 Encoding encoding; | |
979 ASSERT_TRUE(GetDicomEncoding(encoding, value)); | |
980 ASSERT_EQ(expected, encoding); | |
981 } | |
982 | |
983 | |
984 TEST(ParsedDicomFile, DicomMapEncodings1) | |
985 { | |
986 SetDefaultDicomEncoding(Encoding_Ascii); | |
987 ASSERT_EQ(Encoding_Ascii, GetDefaultDicomEncoding()); | |
988 | |
989 { | |
990 DicomMap m; | |
991 ParsedDicomFile dicom(m, GetDefaultDicomEncoding(), false); | |
992 ASSERT_EQ(1u, dicom.GetDcmtkObject().getDataset()->card()); | |
993 CheckEncoding(dicom, Encoding_Ascii); | |
994 } | |
995 | |
996 { | |
997 DicomMap m; | |
998 ParsedDicomFile dicom(m, Encoding_Latin4, false); | |
999 ASSERT_EQ(1u, dicom.GetDcmtkObject().getDataset()->card()); | |
1000 CheckEncoding(dicom, Encoding_Latin4); | |
1001 } | |
1002 | |
1003 { | |
1004 DicomMap m; | |
1005 m.SetValue(DICOM_TAG_SPECIFIC_CHARACTER_SET, "ISO_IR 148", false); | |
1006 ParsedDicomFile dicom(m, GetDefaultDicomEncoding(), false); | |
1007 ASSERT_EQ(1u, dicom.GetDcmtkObject().getDataset()->card()); | |
1008 CheckEncoding(dicom, Encoding_Latin5); | |
1009 } | |
1010 | |
1011 { | |
1012 DicomMap m; | |
1013 m.SetValue(DICOM_TAG_SPECIFIC_CHARACTER_SET, "ISO_IR 148", false); | |
1014 ParsedDicomFile dicom(m, Encoding_Latin1, false); | |
1015 ASSERT_EQ(1u, dicom.GetDcmtkObject().getDataset()->card()); | |
1016 CheckEncoding(dicom, Encoding_Latin5); | |
1017 } | |
1018 } | |
1019 | |
1020 | |
1021 TEST(ParsedDicomFile, DicomMapEncodings2) | |
1022 { | |
1023 const char* utf8 = NULL; | |
1024 for (unsigned int i = 0; i < testEncodingsCount; i++) | |
1025 { | |
1026 if (testEncodings[i] == Encoding_Utf8) | |
1027 { | |
1028 utf8 = testEncodingsEncoded[i]; | |
1029 break; | |
1030 } | |
1031 } | |
1032 | |
1033 ASSERT_TRUE(utf8 != NULL); | |
1034 | |
1035 for (unsigned int i = 0; i < testEncodingsCount; i++) | |
1036 { | |
1037 // 1251 codepage is not supported by the core DICOM standard, ignore it | |
1038 if (testEncodings[i] != Encoding_Windows1251) | |
1039 { | |
1040 { | |
1041 // Sanity check to test the proper behavior of "EncodingTests.py" | |
1042 std::string encoded = Toolbox::ConvertFromUtf8(testEncodingsExpected[i], testEncodings[i]); | |
1043 ASSERT_STREQ(testEncodingsEncoded[i], encoded.c_str()); | |
1044 std::string decoded = Toolbox::ConvertToUtf8(encoded, testEncodings[i], false); | |
1045 ASSERT_STREQ(testEncodingsExpected[i], decoded.c_str()); | |
1046 | |
1047 if (testEncodings[i] != Encoding_Chinese) | |
1048 { | |
1049 // A specific source string is used in "EncodingTests.py" to | |
1050 // test against Chinese, it is normal that it does not correspond to UTF8 | |
1051 | |
1052 std::string encoded = Toolbox::ConvertToUtf8(Toolbox::ConvertFromUtf8(utf8, testEncodings[i]), testEncodings[i], false); | |
1053 ASSERT_STREQ(testEncodingsExpected[i], encoded.c_str()); | |
1054 } | |
1055 } | |
1056 | |
1057 | |
1058 Json::Value v; | |
1059 | |
1060 { | |
1061 DicomMap m; | |
1062 m.SetValue(DICOM_TAG_PATIENT_NAME, testEncodingsExpected[i], false); | |
1063 | |
1064 ParsedDicomFile dicom(m, testEncodings[i], false); | |
1065 | |
1066 const char* encoded = NULL; | |
1067 ASSERT_TRUE(dicom.GetDcmtkObject().getDataset()->findAndGetString(DCM_PatientName, encoded).good()); | |
1068 ASSERT_STREQ(testEncodingsEncoded[i], encoded); | |
1069 | |
1070 dicom.DatasetToJson(v, DicomToJsonFormat_Human, DicomToJsonFlags_Default, 0); | |
1071 | |
1072 Encoding encoding; | |
1073 ASSERT_TRUE(GetDicomEncoding(encoding, v["SpecificCharacterSet"].asCString())); | |
1074 ASSERT_EQ(encoding, testEncodings[i]); | |
1075 ASSERT_STREQ(testEncodingsExpected[i], v["PatientName"].asCString()); | |
1076 } | |
1077 | |
1078 | |
1079 { | |
1080 DicomMap m; | |
1081 m.SetValue(DICOM_TAG_SPECIFIC_CHARACTER_SET, GetDicomSpecificCharacterSet(testEncodings[i]), false); | |
1082 m.SetValue(DICOM_TAG_PATIENT_NAME, testEncodingsExpected[i], false); | |
1083 | |
1084 ParsedDicomFile dicom(m, testEncodings[i], false); | |
1085 | |
1086 Json::Value v2; | |
1087 dicom.DatasetToJson(v2, DicomToJsonFormat_Human, DicomToJsonFlags_Default, 0); | |
1088 | |
1089 ASSERT_EQ(v2["PatientName"].asString(), v["PatientName"].asString()); | |
1090 ASSERT_EQ(v2["SpecificCharacterSet"].asString(), v["SpecificCharacterSet"].asString()); | |
1091 } | |
1092 } | |
1093 } | |
1094 } | |
1095 | |
1096 | |
1097 TEST(ParsedDicomFile, ChangeEncoding) | |
1098 { | |
1099 for (unsigned int i = 0; i < testEncodingsCount; i++) | |
1100 { | |
1101 // 1251 codepage is not supported by the core DICOM standard, ignore it | |
1102 if (testEncodings[i] != Encoding_Windows1251) | |
1103 { | |
1104 DicomMap m; | |
1105 m.SetValue(DICOM_TAG_PATIENT_NAME, testEncodingsExpected[i], false); | |
1106 | |
1107 std::string tag; | |
1108 | |
1109 ParsedDicomFile dicom(m, Encoding_Utf8, false); | |
1110 bool hasCodeExtensions; | |
1111 ASSERT_EQ(Encoding_Utf8, dicom.DetectEncoding(hasCodeExtensions)); | |
1112 ASSERT_FALSE(hasCodeExtensions); | |
1113 ASSERT_TRUE(dicom.GetTagValue(tag, DICOM_TAG_PATIENT_NAME)); | |
1114 ASSERT_EQ(tag, testEncodingsExpected[i]); | |
1115 | |
1116 { | |
1117 Json::Value v; | |
1118 dicom.DatasetToJson(v, DicomToJsonFormat_Human, DicomToJsonFlags_Default, 0); | |
1119 ASSERT_STREQ(v["SpecificCharacterSet"].asCString(), "ISO_IR 192"); | |
1120 ASSERT_STREQ(v["PatientName"].asCString(), testEncodingsExpected[i]); | |
1121 } | |
1122 | |
1123 dicom.ChangeEncoding(testEncodings[i]); | |
1124 | |
1125 ASSERT_EQ(testEncodings[i], dicom.DetectEncoding(hasCodeExtensions)); | |
1126 ASSERT_FALSE(hasCodeExtensions); | |
1127 | |
1128 const char* c = NULL; | |
1129 ASSERT_TRUE(dicom.GetDcmtkObject().getDataset()->findAndGetString(DCM_PatientName, c).good()); | |
1130 EXPECT_STREQ(c, testEncodingsEncoded[i]); | |
1131 | |
1132 ASSERT_TRUE(dicom.GetTagValue(tag, DICOM_TAG_PATIENT_NAME)); // Decodes to UTF-8 | |
1133 EXPECT_EQ(tag, testEncodingsExpected[i]); | |
1134 | |
1135 { | |
1136 Json::Value v; | |
1137 dicom.DatasetToJson(v, DicomToJsonFormat_Human, DicomToJsonFlags_Default, 0); | |
1138 ASSERT_STREQ(v["SpecificCharacterSet"].asCString(), GetDicomSpecificCharacterSet(testEncodings[i])); | |
1139 ASSERT_STREQ(v["PatientName"].asCString(), testEncodingsExpected[i]); | |
1140 } | |
1141 } | |
1142 } | |
1143 } | |
1144 | |
1145 | |
1146 TEST(Toolbox, CaseWithAccents) | |
1147 { | |
1148 ASSERT_EQ(toUpperResult, Toolbox::ToUpperCaseWithAccents(toUpperSource)); | |
1149 } | |
1150 | |
1151 | |
1152 | |
1153 TEST(ParsedDicomFile, InvalidCharacterSets) | |
1154 { | |
1155 { | |
1156 // No encoding provided, fallback to default encoding | |
1157 DicomMap m; | |
1158 m.SetValue(DICOM_TAG_PATIENT_NAME, "HELLO", false); | |
1159 | |
1160 ParsedDicomFile d(m, Encoding_Latin3 /* default encoding */, false); | |
1161 | |
1162 bool hasCodeExtensions; | |
1163 ASSERT_EQ(Encoding_Latin3, d.DetectEncoding(hasCodeExtensions)); | |
1164 ASSERT_FALSE(hasCodeExtensions); | |
1165 } | |
1166 | |
1167 { | |
1168 // Valid encoding, "ISO_IR 13" is Japanese | |
1169 DicomMap m; | |
1170 m.SetValue(DICOM_TAG_SPECIFIC_CHARACTER_SET, "ISO_IR 13", false); | |
1171 m.SetValue(DICOM_TAG_PATIENT_NAME, "HELLO", false); | |
1172 | |
1173 ParsedDicomFile d(m, Encoding_Latin3 /* default encoding */, false); | |
1174 | |
1175 bool hasCodeExtensions; | |
1176 ASSERT_EQ(Encoding_Japanese, d.DetectEncoding(hasCodeExtensions)); | |
1177 ASSERT_FALSE(hasCodeExtensions); | |
1178 } | |
1179 | |
1180 { | |
1181 // Invalid value for an encoding ("nope" is not in the DICOM standard) | |
1182 DicomMap m; | |
1183 m.SetValue(DICOM_TAG_SPECIFIC_CHARACTER_SET, "nope", false); | |
1184 m.SetValue(DICOM_TAG_PATIENT_NAME, "HELLO", false); | |
1185 | |
1186 ASSERT_THROW(ParsedDicomFile d(m, Encoding_Latin3, false), | |
1187 OrthancException); | |
1188 } | |
1189 | |
1190 { | |
1191 // Invalid encoding, as provided as a binary string | |
1192 DicomMap m; | |
1193 m.SetValue(DICOM_TAG_SPECIFIC_CHARACTER_SET, "ISO_IR 13", true); | |
1194 m.SetValue(DICOM_TAG_PATIENT_NAME, "HELLO", false); | |
1195 | |
1196 ASSERT_THROW(ParsedDicomFile d(m, Encoding_Latin3, false), | |
1197 OrthancException); | |
1198 } | |
1199 | |
1200 { | |
1201 // Encoding provided as an empty string, fallback to default encoding | |
1202 // In Orthanc <= 1.3.1, this test was throwing an exception | |
1203 DicomMap m; | |
1204 m.SetValue(DICOM_TAG_SPECIFIC_CHARACTER_SET, "", false); | |
1205 m.SetValue(DICOM_TAG_PATIENT_NAME, "HELLO", false); | |
1206 | |
1207 ParsedDicomFile d(m, Encoding_Latin3 /* default encoding */, false); | |
1208 | |
1209 bool hasCodeExtensions; | |
1210 ASSERT_EQ(Encoding_Latin3, d.DetectEncoding(hasCodeExtensions)); | |
1211 ASSERT_FALSE(hasCodeExtensions); | |
1212 } | |
1213 } | |
1214 | |
1215 | |
1216 | |
1217 TEST(Toolbox, RemoveIso2022EscapeSequences) | |
1218 { | |
1219 // +----------------------------------+ | |
1220 // | one-byte control messages | | |
1221 // +----------------------------------+ | |
1222 | |
1223 static const uint8_t iso2022_cstr_oneByteControl[] = { | |
1224 0x0f, 0x41, | |
1225 0x0e, 0x42, | |
1226 0x8e, 0x1b, 0x4e, 0x43, | |
1227 0x8f, 0x1b, 0x4f, 0x44, | |
1228 0x8e, 0x1b, 0x4a, 0x45, | |
1229 0x8f, 0x1b, 0x4a, 0x46, | |
1230 0x50, 0x51, 0x52, 0x00 | |
1231 }; | |
1232 | |
1233 static const uint8_t iso2022_cstr_oneByteControl_ref[] = { | |
1234 0x41, | |
1235 0x42, | |
1236 0x43, | |
1237 0x44, | |
1238 0x8e, 0x1b, 0x4a, 0x45, | |
1239 0x8f, 0x1b, 0x4a, 0x46, | |
1240 0x50, 0x51, 0x52, 0x00 | |
1241 }; | |
1242 | |
1243 // +----------------------------------+ | |
1244 // | two-byte control messages | | |
1245 // +----------------------------------+ | |
1246 | |
1247 static const uint8_t iso2022_cstr_twoByteControl[] = { | |
1248 0x1b, 0x6e, 0x41, | |
1249 0x1b, 0x6f, 0x42, | |
1250 0x1b, 0x4e, 0x43, | |
1251 0x1b, 0x4f, 0x44, | |
1252 0x1b, 0x7e, 0x45, | |
1253 0x1b, 0x7d, 0x46, | |
1254 0x1b, 0x7c, 0x47, 0x00 | |
1255 }; | |
1256 | |
1257 static const uint8_t iso2022_cstr_twoByteControl_ref[] = { | |
1258 0x41, | |
1259 0x42, | |
1260 0x43, | |
1261 0x44, | |
1262 0x45, | |
1263 0x46, | |
1264 0x47, 0x00 | |
1265 }; | |
1266 | |
1267 // +----------------------------------+ | |
1268 // | various-length escape sequences | | |
1269 // +----------------------------------+ | |
1270 | |
1271 static const uint8_t iso2022_cstr_escapeSequence[] = { | |
1272 0x1b, 0x40, 0x41, // 1b and 40 should not be removed (invalid esc seq) | |
1273 0x1b, 0x50, 0x42, // ditto | |
1274 0x1b, 0x7f, 0x43, // ditto | |
1275 0x1b, 0x21, 0x4a, 0x44, // this will match | |
1276 0x1b, 0x20, 0x21, 0x2f, 0x40, 0x45, // this will match | |
1277 0x1b, 0x20, 0x21, 0x2f, 0x2f, 0x40, 0x46, // this will match too | |
1278 0x1b, 0x20, 0x21, 0x2f, 0x1f, 0x47, 0x48, 0x00 // this will NOT match! | |
1279 }; | |
1280 | |
1281 static const uint8_t iso2022_cstr_escapeSequence_ref[] = { | |
1282 0x1b, 0x40, 0x41, // 1b and 40 should not be removed (invalid esc seq) | |
1283 0x1b, 0x50, 0x42, // ditto | |
1284 0x1b, 0x7f, 0x43, // ditto | |
1285 0x44, // this will match | |
1286 0x45, // this will match | |
1287 0x46, // this will match too | |
1288 0x1b, 0x20, 0x21, 0x2f, 0x1f, 0x47, 0x48, 0x00 // this will NOT match! | |
1289 }; | |
1290 | |
1291 | |
1292 // +----------------------------------+ | |
1293 // | a real-world japanese sample | | |
1294 // +----------------------------------+ | |
1295 | |
1296 static const uint8_t iso2022_cstr_real_ir13[] = { | |
1297 0xd4, 0xcf, 0xc0, 0xde, 0x5e, 0xc0, 0xdb, 0xb3, | |
1298 0x3d, 0x1b, 0x24, 0x42, 0x3b, 0x33, 0x45, 0x44, | |
1299 0x1b, 0x28, 0x4a, 0x5e, 0x1b, 0x24, 0x42, 0x42, | |
1300 0x40, 0x4f, 0x3a, 0x1b, 0x28, 0x4a, 0x3d, 0x1b, | |
1301 0x24, 0x42, 0x24, 0x64, 0x24, 0x5e, 0x24, 0x40, | |
1302 0x1b, 0x28, 0x4a, 0x5e, 0x1b, 0x24, 0x42, 0x24, | |
1303 0x3f, 0x24, 0x6d, 0x24, 0x26, 0x1b, 0x28, 0x4a, 0x00 | |
1304 }; | |
1305 | |
1306 static const uint8_t iso2022_cstr_real_ir13_ref[] = { | |
1307 0xd4, 0xcf, 0xc0, 0xde, 0x5e, 0xc0, 0xdb, 0xb3, | |
1308 0x3d, | |
1309 0x3b, 0x33, 0x45, 0x44, | |
1310 0x5e, | |
1311 0x42, | |
1312 0x40, 0x4f, 0x3a, | |
1313 0x3d, | |
1314 0x24, 0x64, 0x24, 0x5e, 0x24, 0x40, | |
1315 0x5e, | |
1316 0x24, | |
1317 0x3f, 0x24, 0x6d, 0x24, 0x26, 0x00 | |
1318 }; | |
1319 | |
1320 | |
1321 | |
1322 // +----------------------------------+ | |
1323 // | the actual test | | |
1324 // +----------------------------------+ | |
1325 | |
1326 std::string iso2022_str_oneByteControl( | |
1327 reinterpret_cast<const char*>(iso2022_cstr_oneByteControl)); | |
1328 std::string iso2022_str_oneByteControl_ref( | |
1329 reinterpret_cast<const char*>(iso2022_cstr_oneByteControl_ref)); | |
1330 std::string iso2022_str_twoByteControl( | |
1331 reinterpret_cast<const char*>(iso2022_cstr_twoByteControl)); | |
1332 std::string iso2022_str_twoByteControl_ref( | |
1333 reinterpret_cast<const char*>(iso2022_cstr_twoByteControl_ref)); | |
1334 std::string iso2022_str_escapeSequence( | |
1335 reinterpret_cast<const char*>(iso2022_cstr_escapeSequence)); | |
1336 std::string iso2022_str_escapeSequence_ref( | |
1337 reinterpret_cast<const char*>(iso2022_cstr_escapeSequence_ref)); | |
1338 std::string iso2022_str_real_ir13( | |
1339 reinterpret_cast<const char*>(iso2022_cstr_real_ir13)); | |
1340 std::string iso2022_str_real_ir13_ref( | |
1341 reinterpret_cast<const char*>(iso2022_cstr_real_ir13_ref)); | |
1342 | |
1343 std::string dest; | |
1344 | |
1345 Toolbox::RemoveIso2022EscapeSequences(dest, iso2022_str_oneByteControl); | |
1346 ASSERT_EQ(dest, iso2022_str_oneByteControl_ref); | |
1347 | |
1348 Toolbox::RemoveIso2022EscapeSequences(dest, iso2022_str_twoByteControl); | |
1349 ASSERT_EQ(dest, iso2022_str_twoByteControl_ref); | |
1350 | |
1351 Toolbox::RemoveIso2022EscapeSequences(dest, iso2022_str_escapeSequence); | |
1352 ASSERT_EQ(dest, iso2022_str_escapeSequence_ref); | |
1353 | |
1354 Toolbox::RemoveIso2022EscapeSequences(dest, iso2022_str_real_ir13); | |
1355 ASSERT_EQ(dest, iso2022_str_real_ir13_ref); | |
1356 } | |
1357 | |
1358 | |
1359 | |
1360 static std::string DecodeFromSpecification(const std::string& s) | |
1361 { | |
1362 std::vector<std::string> tokens; | |
1363 Toolbox::TokenizeString(tokens, s, ' '); | |
1364 | |
1365 std::string result; | |
1366 result.resize(tokens.size()); | |
1367 | |
1368 for (size_t i = 0; i < tokens.size(); i++) | |
1369 { | |
1370 std::vector<std::string> components; | |
1371 Toolbox::TokenizeString(components, tokens[i], '/'); | |
1372 | |
1373 if (components.size() != 2) | |
1374 { | |
1375 throw; | |
1376 } | |
1377 | |
1378 int a = boost::lexical_cast<int>(components[0]); | |
1379 int b = boost::lexical_cast<int>(components[1]); | |
1380 if (a < 0 || a > 15 || | |
1381 b < 0 || b > 15 || | |
1382 (a == 0 && b == 0)) | |
1383 { | |
1384 throw; | |
1385 } | |
1386 | |
1387 result[i] = static_cast<uint8_t>(a * 16 + b); | |
1388 } | |
1389 | |
1390 return result; | |
1391 } | |
1392 | |
1393 | |
1394 | |
1395 // Compatibility wrapper | |
1396 static pugi::xpath_node SelectNode(const pugi::xml_document& doc, | |
1397 const char* xpath) | |
1398 { | |
1399 #if PUGIXML_VERSION <= 140 | |
1400 return doc.select_single_node(xpath); // Deprecated in pugixml 1.5 | |
1401 #else | |
1402 return doc.select_node(xpath); | |
1403 #endif | |
1404 } | |
1405 | |
1406 | |
1407 TEST(Toolbox, EncodingsKorean) | |
1408 { | |
1409 // http://dicom.nema.org/MEDICAL/dicom/current/output/chtml/part05/sect_I.2.html | |
1410 | |
1411 std::string korean = DecodeFromSpecification( | |
1412 "04/08 06/15 06/14 06/07 05/14 04/07 06/09 06/12 06/04 06/15 06/14 06/07 03/13 " | |
1413 "01/11 02/04 02/09 04/03 15/11 15/03 05/14 01/11 02/04 02/09 04/03 13/01 12/14 " | |
1414 "13/04 13/07 03/13 01/11 02/04 02/09 04/03 12/08 10/11 05/14 01/11 02/04 02/09 " | |
1415 "04/03 11/01 14/06 11/05 11/15"); | |
1416 | |
1417 // This array can be re-generated using command-line: | |
1418 // echo -n "Hong^Gildong=..." | hexdump -v -e '14/1 "0x%02x, "' -e '"\n"' | |
1419 static const uint8_t utf8raw[] = { | |
1420 0x48, 0x6f, 0x6e, 0x67, 0x5e, 0x47, 0x69, 0x6c, 0x64, 0x6f, 0x6e, 0x67, 0x3d, 0xe6, | |
1421 0xb4, 0xaa, 0x5e, 0xe5, 0x90, 0x89, 0xe6, 0xb4, 0x9e, 0x3d, 0xed, 0x99, 0x8d, 0x5e, | |
1422 0xea, 0xb8, 0xb8, 0xeb, 0x8f, 0x99 | |
1423 }; | |
1424 | |
1425 std::string utf8(reinterpret_cast<const char*>(utf8raw), sizeof(utf8raw)); | |
1426 | |
1427 ParsedDicomFile dicom(false); | |
1428 dicom.ReplacePlainString(DICOM_TAG_SPECIFIC_CHARACTER_SET, "\\ISO 2022 IR 149"); | |
1429 ASSERT_TRUE(dicom.GetDcmtkObject().getDataset()->putAndInsertString | |
1430 (DCM_PatientName, korean.c_str(), OFBool(true)).good()); | |
1431 | |
1432 bool hasCodeExtensions; | |
1433 Encoding encoding = dicom.DetectEncoding(hasCodeExtensions); | |
1434 ASSERT_EQ(Encoding_Korean, encoding); | |
1435 ASSERT_TRUE(hasCodeExtensions); | |
1436 | |
1437 std::string value; | |
1438 ASSERT_TRUE(dicom.GetTagValue(value, DICOM_TAG_PATIENT_NAME)); | |
1439 ASSERT_EQ(utf8, value); | |
1440 | |
1441 DicomWebJsonVisitor visitor; | |
1442 dicom.Apply(visitor); | |
1443 ASSERT_EQ(utf8.substr(0, 12), visitor.GetResult()["00100010"]["Value"][0]["Alphabetic"].asString()); | |
1444 ASSERT_EQ(utf8.substr(13, 10), visitor.GetResult()["00100010"]["Value"][0]["Ideographic"].asString()); | |
1445 ASSERT_EQ(utf8.substr(24), visitor.GetResult()["00100010"]["Value"][0]["Phonetic"].asString()); | |
1446 | |
1447 #if ORTHANC_ENABLE_PUGIXML == 1 | |
1448 // http://dicom.nema.org/medical/dicom/current/output/chtml/part18/sect_F.3.html#table_F.3.1-1 | |
1449 std::string xml; | |
1450 visitor.FormatXml(xml); | |
1451 | |
1452 pugi::xml_document doc; | |
1453 doc.load_buffer(xml.c_str(), xml.size()); | |
1454 | |
1455 pugi::xpath_node node = SelectNode(doc, "//NativeDicomModel/DicomAttribute[@tag=\"00080005\"]/Value"); | |
1456 ASSERT_STREQ("ISO 2022 IR 149", node.node().text().as_string()); | |
1457 | |
1458 node = SelectNode(doc, "//NativeDicomModel/DicomAttribute[@tag=\"00080005\"]"); | |
1459 ASSERT_STREQ("CS", node.node().attribute("vr").value()); | |
1460 | |
1461 node = SelectNode(doc, "//NativeDicomModel/DicomAttribute[@tag=\"00100010\"]"); | |
1462 ASSERT_STREQ("PN", node.node().attribute("vr").value()); | |
1463 | |
1464 node = SelectNode(doc, "//NativeDicomModel/DicomAttribute[@tag=\"00100010\"]/PersonName/Alphabetic/FamilyName"); | |
1465 ASSERT_STREQ("Hong", node.node().text().as_string()); | |
1466 | |
1467 node = SelectNode(doc, "//NativeDicomModel/DicomAttribute[@tag=\"00100010\"]/PersonName/Alphabetic/GivenName"); | |
1468 ASSERT_STREQ("Gildong", node.node().text().as_string()); | |
1469 | |
1470 node = SelectNode(doc, "//NativeDicomModel/DicomAttribute[@tag=\"00100010\"]/PersonName/Ideographic/FamilyName"); | |
1471 ASSERT_EQ(utf8.substr(13, 3), node.node().text().as_string()); | |
1472 | |
1473 node = SelectNode(doc, "//NativeDicomModel/DicomAttribute[@tag=\"00100010\"]/PersonName/Ideographic/GivenName"); | |
1474 ASSERT_EQ(utf8.substr(17, 6), node.node().text().as_string()); | |
1475 | |
1476 node = SelectNode(doc, "//NativeDicomModel/DicomAttribute[@tag=\"00100010\"]/PersonName/Phonetic/FamilyName"); | |
1477 ASSERT_EQ(utf8.substr(24, 3), node.node().text().as_string()); | |
1478 | |
1479 node = SelectNode(doc, "//NativeDicomModel/DicomAttribute[@tag=\"00100010\"]/PersonName/Phonetic/GivenName"); | |
1480 ASSERT_EQ(utf8.substr(28), node.node().text().as_string()); | |
1481 #endif | |
1482 | |
1483 { | |
1484 DicomMap m; | |
1485 m.FromDicomWeb(visitor.GetResult()); | |
1486 ASSERT_EQ(2u, m.GetSize()); | |
1487 | |
1488 std::string s; | |
1489 ASSERT_TRUE(m.LookupStringValue(s, DICOM_TAG_SPECIFIC_CHARACTER_SET, false)); | |
1490 ASSERT_EQ("ISO 2022 IR 149", s); | |
1491 | |
1492 ASSERT_TRUE(m.LookupStringValue(s, DICOM_TAG_PATIENT_NAME, false)); | |
1493 std::vector<std::string> v; | |
1494 Toolbox::TokenizeString(v, s, '='); | |
1495 ASSERT_EQ(3u, v.size()); | |
1496 ASSERT_EQ("Hong^Gildong", v[0]); | |
1497 ASSERT_EQ(utf8, s); | |
1498 } | |
1499 } | |
1500 | |
1501 | |
1502 TEST(Toolbox, EncodingsJapaneseKanji) | |
1503 { | |
1504 // http://dicom.nema.org/MEDICAL/dicom/current/output/chtml/part05/sect_H.3.html | |
1505 | |
1506 std::string japanese = DecodeFromSpecification( | |
1507 "05/09 06/01 06/13 06/01 06/04 06/01 05/14 05/04 06/01 07/02 06/15 07/05 03/13 " | |
1508 "01/11 02/04 04/02 03/11 03/03 04/05 04/04 01/11 02/08 04/02 05/14 01/11 02/04 " | |
1509 "04/02 04/02 04/00 04/15 03/10 01/11 02/08 04/02 03/13 01/11 02/04 04/02 02/04 " | |
1510 "06/04 02/04 05/14 02/04 04/00 01/11 02/08 04/02 05/14 01/11 02/04 04/02 02/04 " | |
1511 "03/15 02/04 06/13 02/04 02/06 01/11 02/08 04/02"); | |
1512 | |
1513 // This array can be re-generated using command-line: | |
1514 // echo -n "Yamada^Tarou=..." | hexdump -v -e '14/1 "0x%02x, "' -e '"\n"' | |
1515 static const uint8_t utf8raw[] = { | |
1516 0x59, 0x61, 0x6d, 0x61, 0x64, 0x61, 0x5e, 0x54, 0x61, 0x72, 0x6f, 0x75, 0x3d, 0xe5, | |
1517 0xb1, 0xb1, 0xe7, 0x94, 0xb0, 0x5e, 0xe5, 0xa4, 0xaa, 0xe9, 0x83, 0x8e, 0x3d, 0xe3, | |
1518 0x82, 0x84, 0xe3, 0x81, 0xbe, 0xe3, 0x81, 0xa0, 0x5e, 0xe3, 0x81, 0x9f, 0xe3, 0x82, | |
1519 0x8d, 0xe3, 0x81, 0x86 | |
1520 }; | |
1521 | |
1522 std::string utf8(reinterpret_cast<const char*>(utf8raw), sizeof(utf8raw)); | |
1523 | |
1524 ParsedDicomFile dicom(false); | |
1525 dicom.ReplacePlainString(DICOM_TAG_SPECIFIC_CHARACTER_SET, "\\ISO 2022 IR 87"); | |
1526 ASSERT_TRUE(dicom.GetDcmtkObject().getDataset()->putAndInsertString | |
1527 (DCM_PatientName, japanese.c_str(), OFBool(true)).good()); | |
1528 | |
1529 bool hasCodeExtensions; | |
1530 Encoding encoding = dicom.DetectEncoding(hasCodeExtensions); | |
1531 ASSERT_EQ(Encoding_JapaneseKanji, encoding); | |
1532 ASSERT_TRUE(hasCodeExtensions); | |
1533 | |
1534 std::string value; | |
1535 ASSERT_TRUE(dicom.GetTagValue(value, DICOM_TAG_PATIENT_NAME)); | |
1536 ASSERT_EQ(utf8, value); | |
1537 | |
1538 DicomWebJsonVisitor visitor; | |
1539 dicom.Apply(visitor); | |
1540 ASSERT_EQ(utf8.substr(0, 12), visitor.GetResult()["00100010"]["Value"][0]["Alphabetic"].asString()); | |
1541 ASSERT_EQ(utf8.substr(13, 13), visitor.GetResult()["00100010"]["Value"][0]["Ideographic"].asString()); | |
1542 ASSERT_EQ(utf8.substr(27), visitor.GetResult()["00100010"]["Value"][0]["Phonetic"].asString()); | |
1543 | |
1544 #if ORTHANC_ENABLE_PUGIXML == 1 | |
1545 // http://dicom.nema.org/medical/dicom/current/output/chtml/part18/sect_F.3.html#table_F.3.1-1 | |
1546 std::string xml; | |
1547 visitor.FormatXml(xml); | |
1548 | |
1549 pugi::xml_document doc; | |
1550 doc.load_buffer(xml.c_str(), xml.size()); | |
1551 | |
1552 pugi::xpath_node node = SelectNode(doc, "//NativeDicomModel/DicomAttribute[@tag=\"00080005\"]/Value"); | |
1553 ASSERT_STREQ("ISO 2022 IR 87", node.node().text().as_string()); | |
1554 | |
1555 node = SelectNode(doc, "//NativeDicomModel/DicomAttribute[@tag=\"00080005\"]"); | |
1556 ASSERT_STREQ("CS", node.node().attribute("vr").value()); | |
1557 | |
1558 node = SelectNode(doc, "//NativeDicomModel/DicomAttribute[@tag=\"00100010\"]"); | |
1559 ASSERT_STREQ("PN", node.node().attribute("vr").value()); | |
1560 | |
1561 node = SelectNode(doc, "//NativeDicomModel/DicomAttribute[@tag=\"00100010\"]/PersonName/Alphabetic/FamilyName"); | |
1562 ASSERT_STREQ("Yamada", node.node().text().as_string()); | |
1563 | |
1564 node = SelectNode(doc, "//NativeDicomModel/DicomAttribute[@tag=\"00100010\"]/PersonName/Alphabetic/GivenName"); | |
1565 ASSERT_STREQ("Tarou", node.node().text().as_string()); | |
1566 | |
1567 node = SelectNode(doc, "//NativeDicomModel/DicomAttribute[@tag=\"00100010\"]/PersonName/Ideographic/FamilyName"); | |
1568 ASSERT_EQ(utf8.substr(13, 6), node.node().text().as_string()); | |
1569 | |
1570 node = SelectNode(doc, "//NativeDicomModel/DicomAttribute[@tag=\"00100010\"]/PersonName/Ideographic/GivenName"); | |
1571 ASSERT_EQ(utf8.substr(20, 6), node.node().text().as_string()); | |
1572 | |
1573 node = SelectNode(doc, "//NativeDicomModel/DicomAttribute[@tag=\"00100010\"]/PersonName/Phonetic/FamilyName"); | |
1574 ASSERT_EQ(utf8.substr(27, 9), node.node().text().as_string()); | |
1575 | |
1576 node = SelectNode(doc, "//NativeDicomModel/DicomAttribute[@tag=\"00100010\"]/PersonName/Phonetic/GivenName"); | |
1577 ASSERT_EQ(utf8.substr(37), node.node().text().as_string()); | |
1578 #endif | |
1579 | |
1580 { | |
1581 DicomMap m; | |
1582 m.FromDicomWeb(visitor.GetResult()); | |
1583 ASSERT_EQ(2u, m.GetSize()); | |
1584 | |
1585 std::string s; | |
1586 ASSERT_TRUE(m.LookupStringValue(s, DICOM_TAG_SPECIFIC_CHARACTER_SET, false)); | |
1587 ASSERT_EQ("ISO 2022 IR 87", s); | |
1588 | |
1589 ASSERT_TRUE(m.LookupStringValue(s, DICOM_TAG_PATIENT_NAME, false)); | |
1590 std::vector<std::string> v; | |
1591 Toolbox::TokenizeString(v, s, '='); | |
1592 ASSERT_EQ(3u, v.size()); | |
1593 ASSERT_EQ("Yamada^Tarou", v[0]); | |
1594 ASSERT_EQ(utf8, s); | |
1595 } | |
1596 } | |
1597 | |
1598 | |
1599 | |
1600 TEST(Toolbox, EncodingsChinese3) | |
1601 { | |
1602 // http://dicom.nema.org/MEDICAL/dicom/current/output/chtml/part05/sect_J.3.html | |
1603 | |
1604 static const uint8_t chinese[] = { | |
1605 0x57, 0x61, 0x6e, 0x67, 0x5e, 0x58, 0x69, 0x61, 0x6f, 0x44, 0x6f, | |
1606 0x6e, 0x67, 0x3d, 0xcd, 0xf5, 0x5e, 0xd0, 0xa1, 0xb6, 0xab, 0x3d, 0x00 | |
1607 }; | |
1608 | |
1609 ParsedDicomFile dicom(false); | |
1610 dicom.ReplacePlainString(DICOM_TAG_SPECIFIC_CHARACTER_SET, "GB18030"); | |
1611 ASSERT_TRUE(dicom.GetDcmtkObject().getDataset()->putAndInsertString | |
1612 (DCM_PatientName, reinterpret_cast<const char*>(chinese), OFBool(true)).good()); | |
1613 | |
1614 bool hasCodeExtensions; | |
1615 Encoding encoding = dicom.DetectEncoding(hasCodeExtensions); | |
1616 ASSERT_EQ(Encoding_Chinese, encoding); | |
1617 ASSERT_FALSE(hasCodeExtensions); | |
1618 | |
1619 std::string value; | |
1620 ASSERT_TRUE(dicom.GetTagValue(value, DICOM_TAG_PATIENT_NAME)); | |
1621 | |
1622 std::vector<std::string> tokens; | |
1623 Orthanc::Toolbox::TokenizeString(tokens, value, '='); | |
1624 ASSERT_EQ(3u, tokens.size()); | |
1625 ASSERT_EQ("Wang^XiaoDong", tokens[0]); | |
1626 ASSERT_TRUE(tokens[2].empty()); | |
1627 | |
1628 std::vector<std::string> middle; | |
1629 Orthanc::Toolbox::TokenizeString(middle, tokens[1], '^'); | |
1630 ASSERT_EQ(2u, middle.size()); | |
1631 ASSERT_EQ(3u, middle[0].size()); | |
1632 ASSERT_EQ(6u, middle[1].size()); | |
1633 | |
1634 // CDF5 in GB18030 | |
1635 ASSERT_EQ(static_cast<char>(0xe7), middle[0][0]); | |
1636 ASSERT_EQ(static_cast<char>(0x8e), middle[0][1]); | |
1637 ASSERT_EQ(static_cast<char>(0x8b), middle[0][2]); | |
1638 | |
1639 // D0A1 in GB18030 | |
1640 ASSERT_EQ(static_cast<char>(0xe5), middle[1][0]); | |
1641 ASSERT_EQ(static_cast<char>(0xb0), middle[1][1]); | |
1642 ASSERT_EQ(static_cast<char>(0x8f), middle[1][2]); | |
1643 | |
1644 // B6AB in GB18030 | |
1645 ASSERT_EQ(static_cast<char>(0xe4), middle[1][3]); | |
1646 ASSERT_EQ(static_cast<char>(0xb8), middle[1][4]); | |
1647 ASSERT_EQ(static_cast<char>(0x9c), middle[1][5]); | |
1648 } | |
1649 | |
1650 | |
1651 TEST(Toolbox, EncodingsChinese4) | |
1652 { | |
1653 // http://dicom.nema.org/MEDICAL/dicom/current/output/chtml/part05/sect_J.4.html | |
1654 | |
1655 static const uint8_t chinese[] = { | |
1656 0x54, 0x68, 0x65, 0x20, 0x66, 0x69, 0x72, 0x73, 0x74, 0x20, 0x6c, 0x69, 0x6e, | |
1657 0x65, 0x20, 0x69, 0x6e, 0x63, 0x6c, 0x75, 0x64, 0x65, 0x73, 0xd6, 0xd0, 0xce, | |
1658 0xc4, 0x2e, 0x0d, 0x0a, 0x54, 0x68, 0x65, 0x20, 0x73, 0x65, 0x63, 0x6f, 0x6e, | |
1659 0x64, 0x20, 0x6c, 0x69, 0x6e, 0x65, 0x20, 0x69, 0x6e, 0x63, 0x6c, 0x75, 0x64, | |
1660 0x65, 0x73, 0xd6, 0xd0, 0xce, 0xc4, 0x2c, 0x20, 0x74, 0x6f, 0x6f, 0x2e, 0x0d, | |
1661 0x0a, 0x54, 0x68, 0x65, 0x20, 0x74, 0x68, 0x69, 0x72, 0x64, 0x20, 0x6c, 0x69, | |
1662 0x6e, 0x65, 0x2e, 0x0d, 0x0a, 0x00 | |
1663 }; | |
1664 | |
1665 static const uint8_t patternRaw[] = { | |
1666 0xe4, 0xb8, 0xad, 0xe6, 0x96, 0x87 | |
1667 }; | |
1668 | |
1669 const std::string pattern(reinterpret_cast<const char*>(patternRaw), sizeof(patternRaw)); | |
1670 | |
1671 ParsedDicomFile dicom(false); | |
1672 dicom.ReplacePlainString(DICOM_TAG_SPECIFIC_CHARACTER_SET, "GB18030"); | |
1673 ASSERT_TRUE(dicom.GetDcmtkObject().getDataset()->putAndInsertString | |
1674 (DCM_PatientComments, reinterpret_cast<const char*>(chinese), OFBool(true)).good()); | |
1675 | |
1676 bool hasCodeExtensions; | |
1677 Encoding encoding = dicom.DetectEncoding(hasCodeExtensions); | |
1678 ASSERT_EQ(Encoding_Chinese, encoding); | |
1679 ASSERT_FALSE(hasCodeExtensions); | |
1680 | |
1681 std::string value; | |
1682 ASSERT_TRUE(dicom.GetTagValue(value, DICOM_TAG_PATIENT_COMMENTS)); | |
1683 | |
1684 std::vector<std::string> lines; | |
1685 Orthanc::Toolbox::TokenizeString(lines, value, '\n'); | |
1686 ASSERT_EQ(4u, lines.size()); | |
1687 ASSERT_TRUE(boost::starts_with(lines[0], "The first line includes")); | |
1688 ASSERT_TRUE(boost::ends_with(lines[0], ".\r")); | |
1689 ASSERT_TRUE(lines[0].find(pattern) != std::string::npos); | |
1690 ASSERT_TRUE(boost::starts_with(lines[1], "The second line includes")); | |
1691 ASSERT_TRUE(boost::ends_with(lines[1], ", too.\r")); | |
1692 ASSERT_TRUE(lines[1].find(pattern) != std::string::npos); | |
1693 ASSERT_EQ("The third line.\r", lines[2]); | |
1694 ASSERT_FALSE(lines[1].find(pattern) == std::string::npos); | |
1695 ASSERT_TRUE(lines[3].empty()); | |
1696 } | |
1697 | |
1698 | |
1699 TEST(Toolbox, EncodingsSimplifiedChinese2) | |
1700 { | |
1701 // http://dicom.nema.org/MEDICAL/dicom/current/output/chtml/part05/sect_K.2.html | |
1702 | |
1703 static const uint8_t chinese[] = { | |
1704 0x5a, 0x68, 0x61, 0x6e, 0x67, 0x5e, 0x58, 0x69, 0x61, 0x6f, 0x44, 0x6f, | |
1705 0x6e, 0x67, 0x3d, 0x1b, 0x24, 0x29, 0x41, 0xd5, 0xc5, 0x5e, 0x1b, 0x24, | |
1706 0x29, 0x41, 0xd0, 0xa1, 0xb6, 0xab, 0x3d, 0x20, 0x00 | |
1707 }; | |
1708 | |
1709 // echo -n "Zhang^XiaoDong=..." | hexdump -v -e '14/1 "0x%02x, "' -e '"\n"' | |
1710 static const uint8_t utf8[] = { | |
1711 0x5a, 0x68, 0x61, 0x6e, 0x67, 0x5e, 0x58, 0x69, 0x61, 0x6f, 0x44, 0x6f, 0x6e, 0x67, | |
1712 0x3d, 0xe5, 0xbc, 0xa0, 0x5e, 0xe5, 0xb0, 0x8f, 0xe4, 0xb8, 0x9c, 0x3d | |
1713 }; | |
1714 | |
1715 ParsedDicomFile dicom(false); | |
1716 dicom.ReplacePlainString(DICOM_TAG_SPECIFIC_CHARACTER_SET, "\\ISO 2022 IR 58"); | |
1717 ASSERT_TRUE(dicom.GetDcmtkObject().getDataset()->putAndInsertString | |
1718 (DCM_PatientName, reinterpret_cast<const char*>(chinese), OFBool(true)).good()); | |
1719 | |
1720 bool hasCodeExtensions; | |
1721 Encoding encoding = dicom.DetectEncoding(hasCodeExtensions); | |
1722 ASSERT_EQ(Encoding_SimplifiedChinese, encoding); | |
1723 ASSERT_TRUE(hasCodeExtensions); | |
1724 | |
1725 std::string value; | |
1726 ASSERT_TRUE(dicom.GetTagValue(value, DICOM_TAG_PATIENT_NAME)); | |
1727 ASSERT_EQ(value, std::string(reinterpret_cast<const char*>(utf8), sizeof(utf8))); | |
1728 } | |
1729 | |
1730 | |
1731 TEST(Toolbox, EncodingsSimplifiedChinese3) | |
1732 { | |
1733 // http://dicom.nema.org/MEDICAL/dicom/current/output/chtml/part05/sect_K.2.html | |
1734 | |
1735 static const uint8_t chinese[] = { | |
1736 0x31, 0x2e, 0x1b, 0x24, 0x29, 0x41, 0xb5, 0xda, 0xd2, 0xbb, 0xd0, 0xd0, 0xce, 0xc4, 0xd7, 0xd6, 0xa1, 0xa3, 0x0d, 0x0a, | |
1737 0x32, 0x2e, 0x1b, 0x24, 0x29, 0x41, 0xb5, 0xda, 0xb6, 0xfe, 0xd0, 0xd0, 0xce, 0xc4, 0xd7, 0xd6, 0xa1, 0xa3, 0x0d, 0x0a, | |
1738 0x33, 0x2e, 0x1b, 0x24, 0x29, 0x41, 0xb5, 0xda, 0xc8, 0xfd, 0xd0, 0xd0, 0xce, 0xc4, 0xd7, 0xd6, 0xa1, 0xa3, 0x0d, 0x0a, 0x00 | |
1739 }; | |
1740 | |
1741 static const uint8_t line1[] = { | |
1742 0x31, 0x2e, 0xe7, 0xac, 0xac, 0xe4, 0xb8, 0x80, 0xe8, 0xa1, 0x8c, 0xe6, 0x96, 0x87, | |
1743 0xe5, 0xad, 0x97, 0xe3, 0x80, 0x82, '\r' | |
1744 }; | |
1745 | |
1746 static const uint8_t line2[] = { | |
1747 0x32, 0x2e, 0xe7, 0xac, 0xac, 0xe4, 0xba, 0x8c, 0xe8, 0xa1, 0x8c, 0xe6, 0x96, 0x87, | |
1748 0xe5, 0xad, 0x97, 0xe3, 0x80, 0x82, '\r' | |
1749 }; | |
1750 | |
1751 static const uint8_t line3[] = { | |
1752 0x33, 0x2e, 0xe7, 0xac, 0xac, 0xe4, 0xb8, 0x89, 0xe8, 0xa1, 0x8c, 0xe6, 0x96, 0x87, | |
1753 0xe5, 0xad, 0x97, 0xe3, 0x80, 0x82, '\r' | |
1754 }; | |
1755 | |
1756 ParsedDicomFile dicom(false); | |
1757 dicom.ReplacePlainString(DICOM_TAG_SPECIFIC_CHARACTER_SET, "\\ISO 2022 IR 58"); | |
1758 ASSERT_TRUE(dicom.GetDcmtkObject().getDataset()->putAndInsertString | |
1759 (DCM_PatientName, reinterpret_cast<const char*>(chinese), OFBool(true)).good()); | |
1760 | |
1761 bool hasCodeExtensions; | |
1762 Encoding encoding = dicom.DetectEncoding(hasCodeExtensions); | |
1763 ASSERT_EQ(Encoding_SimplifiedChinese, encoding); | |
1764 ASSERT_TRUE(hasCodeExtensions); | |
1765 | |
1766 std::string value; | |
1767 ASSERT_TRUE(dicom.GetTagValue(value, DICOM_TAG_PATIENT_NAME)); | |
1768 | |
1769 std::vector<std::string> lines; | |
1770 Toolbox::TokenizeString(lines, value, '\n'); | |
1771 ASSERT_EQ(4u, lines.size()); | |
1772 ASSERT_EQ(std::string(reinterpret_cast<const char*>(line1), sizeof(line1)), lines[0]); | |
1773 ASSERT_EQ(std::string(reinterpret_cast<const char*>(line2), sizeof(line2)), lines[1]); | |
1774 ASSERT_EQ(std::string(reinterpret_cast<const char*>(line3), sizeof(line3)), lines[2]); | |
1775 ASSERT_TRUE(lines[3].empty()); | |
1776 } | |
1777 | |
1778 | |
1779 | |
1780 | |
1781 #if ORTHANC_ENABLE_DCMTK_TRANSCODING == 1 | |
1782 | |
1783 #include "../Sources/DicomNetworking/DicomStoreUserConnection.h" | |
1784 #include "../Sources/DicomParsing/DcmtkTranscoder.h" | |
1785 | |
1786 TEST(Toto, DISABLED_Transcode3) | |
1787 { | |
1788 DicomAssociationParameters p; | |
1789 p.SetRemotePort(2000); | |
1790 | |
1791 DicomStoreUserConnection scu(p); | |
1792 scu.SetCommonClassesProposed(false); | |
1793 scu.SetRetiredBigEndianProposed(true); | |
1794 | |
1795 DcmtkTranscoder transcoder; | |
1796 | |
1797 for (int j = 0; j < 2; j++) | |
1798 for (int i = 0; i <= DicomTransferSyntax_XML; i++) | |
1799 { | |
1800 DicomTransferSyntax a = (DicomTransferSyntax) i; | |
1801 | |
1802 std::string path = ("/home/jodogne/Subversion/orthanc-tests/Database/TransferSyntaxes/" + | |
1803 std::string(GetTransferSyntaxUid(a)) + ".dcm"); | |
1804 if (Orthanc::SystemToolbox::IsRegularFile(path)) | |
1805 { | |
1806 printf("\n======= %s\n", GetTransferSyntaxUid(a)); | |
1807 | |
1808 std::string source; | |
1809 Orthanc::SystemToolbox::ReadFile(source, path); | |
1810 | |
1811 std::string c, i; | |
1812 try | |
1813 { | |
1814 scu.Transcode(c, i, transcoder, source.c_str(), source.size(), false, "", 0); | |
1815 } | |
1816 catch (OrthancException& e) | |
1817 { | |
1818 if (e.GetErrorCode() == ErrorCode_NotImplemented) | |
1819 { | |
1820 LOG(ERROR) << "cannot transcode " << GetTransferSyntaxUid(a); | |
1821 } | |
1822 else | |
1823 { | |
1824 throw e; | |
1825 } | |
1826 } | |
1827 } | |
1828 } | |
1829 } | |
1830 | |
1831 | |
1832 TEST(Toto, DISABLED_Transcode4) | |
1833 { | |
1834 std::string source; | |
1835 Orthanc::SystemToolbox::ReadFile(source, "/home/jodogne/Subversion/orthanc-tests/Database/KarstenHilbertRF.dcm"); | |
1836 | |
1837 std::unique_ptr<DcmFileFormat> toto(FromDcmtkBridge::LoadFromMemoryBuffer(source.c_str(), source.size())); | |
1838 const std::string sourceUid = IDicomTranscoder::GetSopInstanceUid(*toto); | |
1839 | |
1840 DicomTransferSyntax sourceSyntax; | |
1841 ASSERT_TRUE(FromDcmtkBridge::LookupOrthancTransferSyntax(sourceSyntax, *toto)); | |
1842 | |
1843 DcmtkTranscoder transcoder; | |
1844 | |
1845 for (int i = 0; i <= DicomTransferSyntax_XML; i++) | |
1846 { | |
1847 DicomTransferSyntax a = (DicomTransferSyntax) i; | |
1848 | |
1849 std::set<DicomTransferSyntax> s; | |
1850 s.insert(a); | |
1851 | |
1852 std::string t; | |
1853 | |
1854 IDicomTranscoder::DicomImage source, target; | |
1855 source.AcquireParsed(dynamic_cast<DcmFileFormat*>(toto->clone())); | |
1856 | |
1857 if (!transcoder.Transcode(target, source, s, true)) | |
1858 { | |
1859 printf("**************** CANNOT: [%s] => [%s]\n", | |
1860 GetTransferSyntaxUid(sourceSyntax), GetTransferSyntaxUid(a)); | |
1861 } | |
1862 else | |
1863 { | |
1864 DicomTransferSyntax targetSyntax; | |
1865 ASSERT_TRUE(FromDcmtkBridge::LookupOrthancTransferSyntax(targetSyntax, target.GetParsed())); | |
1866 | |
1867 ASSERT_EQ(targetSyntax, a); | |
1868 bool lossy = (a == DicomTransferSyntax_JPEGProcess1 || | |
1869 a == DicomTransferSyntax_JPEGProcess2_4 || | |
1870 a == DicomTransferSyntax_JPEGLSLossy); | |
1871 | |
1872 printf("SIZE: %lu\n", t.size()); | |
1873 if (sourceUid == IDicomTranscoder::GetSopInstanceUid(target.GetParsed())) | |
1874 { | |
1875 ASSERT_FALSE(lossy); | |
1876 } | |
1877 else | |
1878 { | |
1879 ASSERT_TRUE(lossy); | |
1880 } | |
1881 } | |
1882 } | |
1883 } | |
1884 | |
1885 #endif |