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-2021 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 Lesser General Public License
9  * as published by the Free Software Foundation, either version 3 of
10  * the License, or (at your option) any later version.
11  *
12  * This program is distributed in the hope that it will be useful, but
13  * WITHOUT ANY WARRANTY; without even the implied warranty of
14  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15  * Lesser General Public License for more details.
16  *
17  * You should have received a copy of the GNU Lesser General Public
18  * License along with this program. If not, see
19  * <http://www.gnu.org/licenses/>.
20  **/
21 
22 
23 #if ORTHANC_UNIT_TESTS_LINK_FRAMEWORK == 1
24 // Must be the first to be sure to use the Orthanc framework shared library
25 #  include <OrthancFramework.h>
26 #endif
27 
28 #if !defined(ORTHANC_ENABLE_PUGIXML)
29 #  error ORTHANC_ENABLE_PUGIXML is not defined
30 #endif
31 
32 #include "../Sources/EnumerationDictionary.h"
33 
34 #include <gtest/gtest.h>
35 
36 #include "../Sources/DicomFormat/DicomTag.h"
37 #include "../Sources/HttpServer/HttpToolbox.h"
38 #include "../Sources/Logging.h"
39 #include "../Sources/OrthancException.h"
40 #include "../Sources/Toolbox.h"
41 
42 #if ORTHANC_SANDBOXED != 1
43 #  include "../Sources/FileBuffer.h"
44 #  include "../Sources/MetricsRegistry.h"
45 #  include "../Sources/SystemToolbox.h"
46 #  include "../Sources/TemporaryFile.h"
47 #endif
48 
49 #include <ctype.h>
50 
51 
52 using namespace Orthanc;
53 
54 
TEST(Uuid,Generation)55 TEST(Uuid, Generation)
56 {
57   for (int i = 0; i < 10; i++)
58   {
59     std::string s = Toolbox::GenerateUuid();
60     ASSERT_TRUE(Toolbox::IsUuid(s));
61   }
62 }
63 
TEST(Uuid,Test)64 TEST(Uuid, Test)
65 {
66   ASSERT_FALSE(Toolbox::IsUuid(""));
67   ASSERT_FALSE(Toolbox::IsUuid("012345678901234567890123456789012345"));
68   ASSERT_TRUE(Toolbox::IsUuid("550e8400-e29b-41d4-a716-446655440000"));
69   ASSERT_FALSE(Toolbox::IsUuid("550e8400-e29b-41d4-a716-44665544000_"));
70   ASSERT_FALSE(Toolbox::IsUuid("01234567890123456789012345678901234_"));
71   ASSERT_FALSE(Toolbox::StartsWithUuid("550e8400-e29b-41d4-a716-44665544000"));
72   ASSERT_TRUE(Toolbox::StartsWithUuid("550e8400-e29b-41d4-a716-446655440000"));
73   ASSERT_TRUE(Toolbox::StartsWithUuid("550e8400-e29b-41d4-a716-446655440000 ok"));
74   ASSERT_FALSE(Toolbox::StartsWithUuid("550e8400-e29b-41d4-a716-446655440000ok"));
75 }
76 
TEST(Toolbox,IsSHA1)77 TEST(Toolbox, IsSHA1)
78 {
79   ASSERT_FALSE(Toolbox::IsSHA1(""));
80   ASSERT_FALSE(Toolbox::IsSHA1("01234567890123456789012345678901234567890123"));
81   ASSERT_FALSE(Toolbox::IsSHA1("012345678901234567890123456789012345678901234"));
82   ASSERT_TRUE(Toolbox::IsSHA1("b5ed549f-956400ce-69a8c063-bf5b78be-2732a4b9"));
83 
84   std::string sha = "         b5ed549f-956400ce-69a8c063-bf5b78be-2732a4b9          ";
85   ASSERT_TRUE(Toolbox::IsSHA1(sha));
86   sha[3] = '\0';
87   sha[53] = '\0';
88   ASSERT_TRUE(Toolbox::IsSHA1(sha));
89   sha[40] = '\0';
90   ASSERT_FALSE(Toolbox::IsSHA1(sha));
91   ASSERT_FALSE(Toolbox::IsSHA1("       "));
92 
93   ASSERT_TRUE(Toolbox::IsSHA1("16738bc3-e47ed42a-43ce044c-a3414a45-cb069bd0"));
94 
95   std::string s;
96   Toolbox::ComputeSHA1(s, "The quick brown fox jumps over the lazy dog");
97   ASSERT_TRUE(Toolbox::IsSHA1(s));
98   ASSERT_EQ("2fd4e1c6-7a2d28fc-ed849ee1-bb76e739-1b93eb12", s);
99 
100   ASSERT_FALSE(Toolbox::IsSHA1("b5ed549f-956400ce-69a8c063-bf5b78be-2732a4b_"));
101 }
102 
103 
TEST(ParseGetArguments,Basic)104 TEST(ParseGetArguments, Basic)
105 {
106   HttpToolbox::GetArguments b;
107   HttpToolbox::ParseGetArguments(b, "aaa=baaa&bb=a&aa=c");
108 
109   HttpToolbox::Arguments a;
110   HttpToolbox::CompileGetArguments(a, b);
111 
112   ASSERT_EQ(3u, a.size());
113   ASSERT_EQ(a["aaa"], "baaa");
114   ASSERT_EQ(a["bb"], "a");
115   ASSERT_EQ(a["aa"], "c");
116 }
117 
118 
TEST(ParseGetArguments,BasicEmpty)119 TEST(ParseGetArguments, BasicEmpty)
120 {
121   HttpToolbox::GetArguments b;
122   HttpToolbox::ParseGetArguments(b, "aaa&bb=aa&aa");
123 
124   HttpToolbox::Arguments a;
125   HttpToolbox::CompileGetArguments(a, b);
126 
127   ASSERT_EQ(3u, a.size());
128   ASSERT_EQ(a["aaa"], "");
129   ASSERT_EQ(a["bb"], "aa");
130   ASSERT_EQ(a["aa"], "");
131 }
132 
133 
TEST(ParseGetArguments,Single)134 TEST(ParseGetArguments, Single)
135 {
136   HttpToolbox::GetArguments b;
137   HttpToolbox::ParseGetArguments(b, "aaa=baaa");
138 
139   HttpToolbox::Arguments a;
140   HttpToolbox::CompileGetArguments(a, b);
141 
142   ASSERT_EQ(1u, a.size());
143   ASSERT_EQ(a["aaa"], "baaa");
144 }
145 
146 
TEST(ParseGetArguments,SingleEmpty)147 TEST(ParseGetArguments, SingleEmpty)
148 {
149   HttpToolbox::GetArguments b;
150   HttpToolbox::ParseGetArguments(b, "aaa");
151 
152   HttpToolbox::Arguments a;
153   HttpToolbox::CompileGetArguments(a, b);
154 
155   ASSERT_EQ(1u, a.size());
156   ASSERT_EQ(a["aaa"], "");
157 }
158 
159 
TEST(ParseGetQuery,Test1)160 TEST(ParseGetQuery, Test1)
161 {
162   UriComponents uri;
163   HttpToolbox::GetArguments b;
164   HttpToolbox::ParseGetQuery(uri, b, "/instances/test/world?aaa=baaa&bb=a&aa=c");
165 
166   HttpToolbox::Arguments a;
167   HttpToolbox::CompileGetArguments(a, b);
168 
169   ASSERT_EQ(3u, uri.size());
170   ASSERT_EQ("instances", uri[0]);
171   ASSERT_EQ("test", uri[1]);
172   ASSERT_EQ("world", uri[2]);
173   ASSERT_EQ(3u, a.size());
174   ASSERT_EQ(a["aaa"], "baaa");
175   ASSERT_EQ(a["bb"], "a");
176   ASSERT_EQ(a["aa"], "c");
177 }
178 
179 
TEST(ParseGetQuery,Test2)180 TEST(ParseGetQuery, Test2)
181 {
182   UriComponents uri;
183   HttpToolbox::GetArguments b;
184   HttpToolbox::ParseGetQuery(uri, b, "/instances/test/world");
185 
186   HttpToolbox::Arguments a;
187   HttpToolbox::CompileGetArguments(a, b);
188 
189   ASSERT_EQ(3u, uri.size());
190   ASSERT_EQ("instances", uri[0]);
191   ASSERT_EQ("test", uri[1]);
192   ASSERT_EQ("world", uri[2]);
193   ASSERT_EQ(0u, a.size());
194 }
195 
196 
TEST(Uri,SplitUriComponents)197 TEST(Uri, SplitUriComponents)
198 {
199   UriComponents c, d;
200   Toolbox::SplitUriComponents(c, "/cou/hello/world");
201   ASSERT_EQ(3u, c.size());
202   ASSERT_EQ("cou", c[0]);
203   ASSERT_EQ("hello", c[1]);
204   ASSERT_EQ("world", c[2]);
205 
206   Toolbox::SplitUriComponents(c, "/cou/hello/world/");
207   ASSERT_EQ(3u, c.size());
208   ASSERT_EQ("cou", c[0]);
209   ASSERT_EQ("hello", c[1]);
210   ASSERT_EQ("world", c[2]);
211 
212   Toolbox::SplitUriComponents(c, "/cou/hello/world/a");
213   ASSERT_EQ(4u, c.size());
214   ASSERT_EQ("cou", c[0]);
215   ASSERT_EQ("hello", c[1]);
216   ASSERT_EQ("world", c[2]);
217   ASSERT_EQ("a", c[3]);
218 
219   Toolbox::SplitUriComponents(c, "/");
220   ASSERT_EQ(0u, c.size());
221 
222   Toolbox::SplitUriComponents(c, "/hello");
223   ASSERT_EQ(1u, c.size());
224   ASSERT_EQ("hello", c[0]);
225 
226   Toolbox::SplitUriComponents(c, "/hello/");
227   ASSERT_EQ(1u, c.size());
228   ASSERT_EQ("hello", c[0]);
229 
230   ASSERT_THROW(Toolbox::SplitUriComponents(c, ""), OrthancException);
231   ASSERT_THROW(Toolbox::SplitUriComponents(c, "a"), OrthancException);
232   ASSERT_THROW(Toolbox::SplitUriComponents(c, "/coucou//coucou"), OrthancException);
233 
234   c.clear();
235   c.push_back("test");
236   ASSERT_EQ("/", Toolbox::FlattenUri(c, 10));
237 }
238 
239 
TEST(Uri,Truncate)240 TEST(Uri, Truncate)
241 {
242   UriComponents c, d;
243   Toolbox::SplitUriComponents(c, "/cou/hello/world");
244 
245   Toolbox::TruncateUri(d, c, 0);
246   ASSERT_EQ(3u, d.size());
247   ASSERT_EQ("cou", d[0]);
248   ASSERT_EQ("hello", d[1]);
249   ASSERT_EQ("world", d[2]);
250 
251   Toolbox::TruncateUri(d, c, 1);
252   ASSERT_EQ(2u, d.size());
253   ASSERT_EQ("hello", d[0]);
254   ASSERT_EQ("world", d[1]);
255 
256   Toolbox::TruncateUri(d, c, 2);
257   ASSERT_EQ(1u, d.size());
258   ASSERT_EQ("world", d[0]);
259 
260   Toolbox::TruncateUri(d, c, 3);
261   ASSERT_EQ(0u, d.size());
262 
263   Toolbox::TruncateUri(d, c, 4);
264   ASSERT_EQ(0u, d.size());
265 
266   Toolbox::TruncateUri(d, c, 5);
267   ASSERT_EQ(0u, d.size());
268 }
269 
270 
TEST(Uri,Child)271 TEST(Uri, Child)
272 {
273   UriComponents c1;  Toolbox::SplitUriComponents(c1, "/hello/world");
274   UriComponents c2;  Toolbox::SplitUriComponents(c2, "/hello/hello");
275   UriComponents c3;  Toolbox::SplitUriComponents(c3, "/hello");
276   UriComponents c4;  Toolbox::SplitUriComponents(c4, "/world");
277   UriComponents c5;  Toolbox::SplitUriComponents(c5, "/");
278 
279   ASSERT_TRUE(Toolbox::IsChildUri(c1, c1));
280   ASSERT_FALSE(Toolbox::IsChildUri(c1, c2));
281   ASSERT_FALSE(Toolbox::IsChildUri(c1, c3));
282   ASSERT_FALSE(Toolbox::IsChildUri(c1, c4));
283   ASSERT_FALSE(Toolbox::IsChildUri(c1, c5));
284 
285   ASSERT_FALSE(Toolbox::IsChildUri(c2, c1));
286   ASSERT_TRUE(Toolbox::IsChildUri(c2, c2));
287   ASSERT_FALSE(Toolbox::IsChildUri(c2, c3));
288   ASSERT_FALSE(Toolbox::IsChildUri(c2, c4));
289   ASSERT_FALSE(Toolbox::IsChildUri(c2, c5));
290 
291   ASSERT_TRUE(Toolbox::IsChildUri(c3, c1));
292   ASSERT_TRUE(Toolbox::IsChildUri(c3, c2));
293   ASSERT_TRUE(Toolbox::IsChildUri(c3, c3));
294   ASSERT_FALSE(Toolbox::IsChildUri(c3, c4));
295   ASSERT_FALSE(Toolbox::IsChildUri(c3, c5));
296 
297   ASSERT_FALSE(Toolbox::IsChildUri(c4, c1));
298   ASSERT_FALSE(Toolbox::IsChildUri(c4, c2));
299   ASSERT_FALSE(Toolbox::IsChildUri(c4, c3));
300   ASSERT_TRUE(Toolbox::IsChildUri(c4, c4));
301   ASSERT_FALSE(Toolbox::IsChildUri(c4, c5));
302 
303   ASSERT_TRUE(Toolbox::IsChildUri(c5, c1));
304   ASSERT_TRUE(Toolbox::IsChildUri(c5, c2));
305   ASSERT_TRUE(Toolbox::IsChildUri(c5, c3));
306   ASSERT_TRUE(Toolbox::IsChildUri(c5, c4));
307   ASSERT_TRUE(Toolbox::IsChildUri(c5, c5));
308 }
309 
310 
311 #if ORTHANC_SANDBOXED != 1
TEST(Uri,AutodetectMimeType)312 TEST(Uri, AutodetectMimeType)
313 {
314   ASSERT_EQ(MimeType_Binary, SystemToolbox::AutodetectMimeType("../NOTES"));
315   ASSERT_EQ(MimeType_Binary, SystemToolbox::AutodetectMimeType(""));
316   ASSERT_EQ(MimeType_Binary, SystemToolbox::AutodetectMimeType("/"));
317   ASSERT_EQ(MimeType_Binary, SystemToolbox::AutodetectMimeType("a/a"));
318   ASSERT_EQ(MimeType_Binary, SystemToolbox::AutodetectMimeType("..\\a\\"));
319   ASSERT_EQ(MimeType_Binary, SystemToolbox::AutodetectMimeType("..\\a\\a"));
320 
321   ASSERT_EQ(MimeType_PlainText, SystemToolbox::AutodetectMimeType("../NOTES.txt"));
322   ASSERT_EQ(MimeType_PlainText, SystemToolbox::AutodetectMimeType("../coucou.xml/NOTES.txt"));
323   ASSERT_EQ(MimeType_Xml, SystemToolbox::AutodetectMimeType("..\\coucou.\\NOTES.xml"));
324   ASSERT_EQ(MimeType_Xml, SystemToolbox::AutodetectMimeType("../.xml"));
325   ASSERT_EQ(MimeType_Xml, SystemToolbox::AutodetectMimeType("../.XmL"));
326 
327   ASSERT_EQ(MimeType_JavaScript, SystemToolbox::AutodetectMimeType("NOTES.js"));
328   ASSERT_EQ(MimeType_Json, SystemToolbox::AutodetectMimeType("NOTES.json"));
329   ASSERT_EQ(MimeType_Pdf, SystemToolbox::AutodetectMimeType("NOTES.pdf"));
330   ASSERT_EQ(MimeType_Css, SystemToolbox::AutodetectMimeType("NOTES.css"));
331   ASSERT_EQ(MimeType_Html, SystemToolbox::AutodetectMimeType("NOTES.html"));
332   ASSERT_EQ(MimeType_PlainText, SystemToolbox::AutodetectMimeType("NOTES.txt"));
333   ASSERT_EQ(MimeType_Xml, SystemToolbox::AutodetectMimeType("NOTES.xml"));
334   ASSERT_EQ(MimeType_Gif, SystemToolbox::AutodetectMimeType("NOTES.gif"));
335   ASSERT_EQ(MimeType_Jpeg, SystemToolbox::AutodetectMimeType("NOTES.jpg"));
336   ASSERT_EQ(MimeType_Jpeg, SystemToolbox::AutodetectMimeType("NOTES.jpeg"));
337   ASSERT_EQ(MimeType_Png, SystemToolbox::AutodetectMimeType("NOTES.png"));
338   ASSERT_EQ(MimeType_NaCl, SystemToolbox::AutodetectMimeType("NOTES.nexe"));
339   ASSERT_EQ(MimeType_Json, SystemToolbox::AutodetectMimeType("NOTES.nmf"));
340   ASSERT_EQ(MimeType_PNaCl, SystemToolbox::AutodetectMimeType("NOTES.pexe"));
341   ASSERT_EQ(MimeType_Svg, SystemToolbox::AutodetectMimeType("NOTES.svg"));
342   ASSERT_EQ(MimeType_Woff, SystemToolbox::AutodetectMimeType("NOTES.woff"));
343   ASSERT_EQ(MimeType_Woff2, SystemToolbox::AutodetectMimeType("NOTES.woff2"));
344 
345   // Test primitives from the "RegisterDefaultExtensions()" that was
346   // present in the sample "Serve Folders plugin" of Orthanc 1.4.2
347   ASSERT_STREQ("application/javascript", EnumerationToString(SystemToolbox::AutodetectMimeType(".js")));
348   ASSERT_STREQ("application/json", EnumerationToString(SystemToolbox::AutodetectMimeType(".json")));
349   ASSERT_STREQ("application/json", EnumerationToString(SystemToolbox::AutodetectMimeType(".nmf")));
350   ASSERT_STREQ("application/octet-stream", EnumerationToString(SystemToolbox::AutodetectMimeType("")));
351   ASSERT_STREQ("application/wasm", EnumerationToString(SystemToolbox::AutodetectMimeType(".wasm")));
352   ASSERT_STREQ("application/x-font-woff", EnumerationToString(SystemToolbox::AutodetectMimeType(".woff")));
353   ASSERT_STREQ("application/x-nacl", EnumerationToString(SystemToolbox::AutodetectMimeType(".nexe")));
354   ASSERT_STREQ("application/x-pnacl", EnumerationToString(SystemToolbox::AutodetectMimeType(".pexe")));
355   ASSERT_STREQ("application/xml", EnumerationToString(SystemToolbox::AutodetectMimeType(".xml")));
356   ASSERT_STREQ("font/woff2", EnumerationToString(SystemToolbox::AutodetectMimeType(".woff2")));
357   ASSERT_STREQ("image/gif", EnumerationToString(SystemToolbox::AutodetectMimeType(".gif")));
358   ASSERT_STREQ("image/jpeg", EnumerationToString(SystemToolbox::AutodetectMimeType(".jpeg")));
359   ASSERT_STREQ("image/jpeg", EnumerationToString(SystemToolbox::AutodetectMimeType(".jpg")));
360   ASSERT_STREQ("image/png", EnumerationToString(SystemToolbox::AutodetectMimeType(".png")));
361   ASSERT_STREQ("image/svg+xml", EnumerationToString(SystemToolbox::AutodetectMimeType(".svg")));
362   ASSERT_STREQ("text/css", EnumerationToString(SystemToolbox::AutodetectMimeType(".css")));
363   ASSERT_STREQ("text/html", EnumerationToString(SystemToolbox::AutodetectMimeType(".html")));
364 }
365 #endif
366 
367 
TEST(Toolbox,ComputeMD5)368 TEST(Toolbox, ComputeMD5)
369 {
370   std::string s;
371 
372   // # echo -n "Hello" | md5sum
373 
374   Toolbox::ComputeMD5(s, "Hello");
375   ASSERT_EQ("8b1a9953c4611296a827abf8c47804d7", s);
376   Toolbox::ComputeMD5(s, "");
377   ASSERT_EQ("d41d8cd98f00b204e9800998ecf8427e", s);
378 }
379 
TEST(Toolbox,ComputeSHA1)380 TEST(Toolbox, ComputeSHA1)
381 {
382   std::string s;
383 
384   Toolbox::ComputeSHA1(s, "The quick brown fox jumps over the lazy dog");
385   ASSERT_EQ("2fd4e1c6-7a2d28fc-ed849ee1-bb76e739-1b93eb12", s);
386   Toolbox::ComputeSHA1(s, "");
387   ASSERT_EQ("da39a3ee-5e6b4b0d-3255bfef-95601890-afd80709", s);
388 }
389 
390 #if ORTHANC_SANDBOXED != 1
TEST(Toolbox,PathToExecutable)391 TEST(Toolbox, PathToExecutable)
392 {
393   printf("[%s]\n", SystemToolbox::GetPathToExecutable().c_str());
394   printf("[%s]\n", SystemToolbox::GetDirectoryOfExecutable().c_str());
395 }
396 #endif
397 
TEST(Toolbox,StripSpaces)398 TEST(Toolbox, StripSpaces)
399 {
400   ASSERT_EQ("", Toolbox::StripSpaces("       \t  \r   \n  "));
401   ASSERT_EQ("coucou", Toolbox::StripSpaces("    coucou   \t  \r   \n  "));
402   ASSERT_EQ("cou   cou", Toolbox::StripSpaces("    cou   cou    \n  "));
403   ASSERT_EQ("c", Toolbox::StripSpaces("    \n\t c\r    \n  "));
404 
405   std::string s = "\"  abd \"";
406   Toolbox::RemoveSurroundingQuotes(s); ASSERT_EQ("  abd ", s);
407 
408   s = "  \"  abd \"  ";
409   Toolbox::RemoveSurroundingQuotes(s); ASSERT_EQ("  \"  abd \"  ", s);
410 
411   s = Toolbox::StripSpaces(s);
412   Toolbox::RemoveSurroundingQuotes(s); ASSERT_EQ("  abd ", s);
413 
414   s = "\"";
415   Toolbox::RemoveSurroundingQuotes(s); ASSERT_EQ("", s);
416 
417   s = "\"\"";
418   Toolbox::RemoveSurroundingQuotes(s); ASSERT_EQ("", s);
419 
420   s = "\"_\"";
421   Toolbox::RemoveSurroundingQuotes(s); ASSERT_EQ("_", s);
422 
423   s = "\"\"\"";
424   Toolbox::RemoveSurroundingQuotes(s); ASSERT_EQ("\"", s);
425 }
426 
TEST(Toolbox,Case)427 TEST(Toolbox, Case)
428 {
429   std::string s = "CoU";
430   std::string ss;
431 
432   Toolbox::ToUpperCase(ss, s);
433   ASSERT_EQ("COU", ss);
434   Toolbox::ToLowerCase(ss, s);
435   ASSERT_EQ("cou", ss);
436 
437   s = "CoU";
438   Toolbox::ToUpperCase(s);
439   ASSERT_EQ("COU", s);
440 
441   s = "CoU";
442   Toolbox::ToLowerCase(s);
443   ASSERT_EQ("cou", s);
444 }
445 
446 
TEST(Logger,Basic)447 TEST(Logger, Basic)
448 {
449   LOG(INFO) << "I say hello";
450 }
451 
TEST(Toolbox,ConvertFromLatin1)452 TEST(Toolbox, ConvertFromLatin1)
453 {
454   // This is a Latin-1 test string
455   const unsigned char data[10] = { 0xe0, 0xe9, 0xea, 0xe7, 0x26, 0xc6, 0x61, 0x62, 0x63, 0x00 };
456 
457   std::string s((char*) &data[0], 10);
458   ASSERT_EQ("&abc", Toolbox::ConvertToAscii(s));
459 
460   // Open in Emacs, then save with UTF-8 encoding, then "hexdump -C"
461   std::string utf8 = Toolbox::ConvertToUtf8(s, Encoding_Latin1, false);
462   ASSERT_EQ(15u, utf8.size());
463   ASSERT_EQ(0xc3, static_cast<unsigned char>(utf8[0]));
464   ASSERT_EQ(0xa0, static_cast<unsigned char>(utf8[1]));
465   ASSERT_EQ(0xc3, static_cast<unsigned char>(utf8[2]));
466   ASSERT_EQ(0xa9, static_cast<unsigned char>(utf8[3]));
467   ASSERT_EQ(0xc3, static_cast<unsigned char>(utf8[4]));
468   ASSERT_EQ(0xaa, static_cast<unsigned char>(utf8[5]));
469   ASSERT_EQ(0xc3, static_cast<unsigned char>(utf8[6]));
470   ASSERT_EQ(0xa7, static_cast<unsigned char>(utf8[7]));
471   ASSERT_EQ(0x26, static_cast<unsigned char>(utf8[8]));
472   ASSERT_EQ(0xc3, static_cast<unsigned char>(utf8[9]));
473   ASSERT_EQ(0x86, static_cast<unsigned char>(utf8[10]));
474   ASSERT_EQ(0x61, static_cast<unsigned char>(utf8[11]));
475   ASSERT_EQ(0x62, static_cast<unsigned char>(utf8[12]));
476   ASSERT_EQ(0x63, static_cast<unsigned char>(utf8[13]));
477   ASSERT_EQ(0x00, static_cast<unsigned char>(utf8[14]));  // Null-terminated string
478 }
479 
480 
TEST(Toolbox,FixUtf8)481 TEST(Toolbox, FixUtf8)
482 {
483   // This is a Latin-1 test string: "crane" with a circumflex accent
484   const unsigned char latin1[] = { 0x63, 0x72, 0xe2, 0x6e, 0x65 };
485 
486   std::string s((char*) &latin1[0], sizeof(latin1) / sizeof(char));
487 
488   ASSERT_EQ(s, Toolbox::ConvertFromUtf8(Toolbox::ConvertToUtf8(s, Encoding_Latin1, false), Encoding_Latin1));
489   ASSERT_EQ("cre", Toolbox::ConvertToUtf8(s, Encoding_Utf8, false));
490 }
491 
492 
GetUnicode(const uint8_t * data,size_t size,size_t expectedLength)493 static int32_t GetUnicode(const uint8_t* data,
494                           size_t size,
495                           size_t expectedLength)
496 {
497   std::string s((char*) &data[0], size);
498   uint32_t unicode;
499   size_t length;
500   Toolbox::Utf8ToUnicodeCharacter(unicode, length, s, 0);
501   if (length != expectedLength)
502   {
503     return -1;  // Error case
504   }
505   else
506   {
507     return unicode;
508   }
509 }
510 
511 
TEST(Toolbox,Utf8ToUnicode)512 TEST(Toolbox, Utf8ToUnicode)
513 {
514   // https://en.wikipedia.org/wiki/UTF-8
515 
516   ASSERT_EQ(1u, sizeof(char));
517   ASSERT_EQ(1u, sizeof(uint8_t));
518 
519   {
520     const uint8_t data[] = { 0x24 };
521     ASSERT_EQ(0x24, GetUnicode(data, 1, 1));
522     ASSERT_THROW(GetUnicode(data, 0, 1), OrthancException);
523   }
524 
525   {
526     const uint8_t data[] = { 0xc2, 0xa2 };
527     ASSERT_EQ(0xa2, GetUnicode(data, 2, 2));
528     ASSERT_THROW(GetUnicode(data, 1, 2), OrthancException);
529   }
530 
531   {
532     const uint8_t data[] = { 0xe0, 0xa4, 0xb9 };
533     ASSERT_EQ(0x0939, GetUnicode(data, 3, 3));
534     ASSERT_THROW(GetUnicode(data, 2, 3), OrthancException);
535   }
536 
537   {
538     const uint8_t data[] = { 0xe2, 0x82, 0xac };
539     ASSERT_EQ(0x20ac, GetUnicode(data, 3, 3));
540     ASSERT_THROW(GetUnicode(data, 2, 3), OrthancException);
541   }
542 
543   {
544     const uint8_t data[] = { 0xf0, 0x90, 0x8d, 0x88 };
545     ASSERT_EQ(0x010348, GetUnicode(data, 4, 4));
546     ASSERT_THROW(GetUnicode(data, 3, 4), OrthancException);
547   }
548 
549   {
550     const uint8_t data[] = { 0xe0 };
551     ASSERT_THROW(GetUnicode(data, 1, 1), OrthancException);
552   }
553 }
554 
555 
TEST(Toolbox,UrlDecode)556 TEST(Toolbox, UrlDecode)
557 {
558   std::string s;
559 
560   s = "Hello%20World";
561   Toolbox::UrlDecode(s);
562   ASSERT_EQ("Hello World", s);
563 
564   s = "%21%23%24%26%27%28%29%2A%2B%2c%2f%3A%3b%3d%3f%40%5B%5D%90%ff";
565   Toolbox::UrlDecode(s);
566   std::string ss = "!#$&'()*+,/:;=?@[]";
567   ss.push_back((char) 144);
568   ss.push_back((char) 255);
569   ASSERT_EQ(ss, s);
570 
571   s = "(2000%2C00A4)+Other";
572   Toolbox::UrlDecode(s);
573   ASSERT_EQ("(2000,00A4) Other", s);
574 }
575 
576 
TEST(Toolbox,IsAsciiString)577 TEST(Toolbox, IsAsciiString)
578 {
579   std::string s = "Hello 12 /";
580   ASSERT_EQ(10u, s.size());
581   ASSERT_TRUE(Toolbox::IsAsciiString(s));
582   ASSERT_TRUE(Toolbox::IsAsciiString(s.c_str(), 10));
583   ASSERT_FALSE(Toolbox::IsAsciiString(s.c_str(), 11));  // Taking the trailing hidden '\0'
584 
585   s[2] = '\0';
586   ASSERT_EQ(10u, s.size());
587   ASSERT_FALSE(Toolbox::IsAsciiString(s));
588 
589   ASSERT_TRUE(Toolbox::IsAsciiString("Hello\nworld"));
590   ASSERT_FALSE(Toolbox::IsAsciiString("Hello\rworld"));
591 
592   ASSERT_EQ("Hello\nworld", Toolbox::ConvertToAscii("Hello\nworld"));
593   ASSERT_EQ("Helloworld", Toolbox::ConvertToAscii("Hello\r\tworld"));
594 }
595 
596 
597 #if defined(__linux__)
TEST(Toolbox,AbsoluteDirectory)598 TEST(Toolbox, AbsoluteDirectory)
599 {
600   ASSERT_EQ("/tmp/hello", SystemToolbox::InterpretRelativePath("/tmp", "hello"));
601   ASSERT_EQ("/tmp", SystemToolbox::InterpretRelativePath("/tmp", "/tmp"));
602 }
603 #endif
604 
605 
606 #if ORTHANC_SANDBOXED != 1
TEST(Toolbox,WriteFile)607 TEST(Toolbox, WriteFile)
608 {
609   std::string path;
610 
611   {
612     TemporaryFile tmp;
613     path = tmp.GetPath();
614 
615     std::string s;
616     s.append("Hello");
617     s.push_back('\0');
618     s.append("World");
619     ASSERT_EQ(11u, s.size());
620 
621     SystemToolbox::WriteFile(s, path.c_str());
622 
623     std::string t;
624     SystemToolbox::ReadFile(t, path.c_str());
625 
626     ASSERT_EQ(11u, t.size());
627     ASSERT_EQ(0, t[5]);
628     ASSERT_EQ(0, memcmp(s.c_str(), t.c_str(), s.size()));
629 
630     std::string h;
631     ASSERT_EQ(true, SystemToolbox::ReadHeader(h, path.c_str(), 1));
632     ASSERT_EQ(1u, h.size());
633     ASSERT_EQ('H', h[0]);
634     ASSERT_TRUE(SystemToolbox::ReadHeader(h, path.c_str(), 0));
635     ASSERT_EQ(0u, h.size());
636     ASSERT_FALSE(SystemToolbox::ReadHeader(h, path.c_str(), 32));
637     ASSERT_EQ(11u, h.size());
638     ASSERT_EQ(0, memcmp(s.c_str(), h.c_str(), s.size()));
639   }
640 
641   std::string u;
642   ASSERT_THROW(SystemToolbox::ReadFile(u, path.c_str()), OrthancException);
643 
644   {
645     TemporaryFile tmp;
646     std::string s = "Hello";
647     SystemToolbox::WriteFile(s, tmp.GetPath(), true /* call fsync() */);
648     std::string t;
649     SystemToolbox::ReadFile(t, tmp.GetPath());
650     ASSERT_EQ(s, t);
651   }
652 }
653 #endif
654 
655 
656 #if ORTHANC_SANDBOXED != 1
TEST(Toolbox,FileBuffer)657 TEST(Toolbox, FileBuffer)
658 {
659   FileBuffer f;
660   f.Append("a", 1);
661   f.Append("", 0);
662   f.Append("bc", 2);
663 
664   std::string s;
665   f.Read(s);
666   ASSERT_EQ("abc", s);
667 
668   ASSERT_THROW(f.Append("d", 1), OrthancException);  // File is closed
669 }
670 #endif
671 
672 
TEST(Toolbox,Wildcard)673 TEST(Toolbox, Wildcard)
674 {
675   ASSERT_EQ("abcd", Toolbox::WildcardToRegularExpression("abcd"));
676   ASSERT_EQ("ab.*cd", Toolbox::WildcardToRegularExpression("ab*cd"));
677   ASSERT_EQ("ab..cd", Toolbox::WildcardToRegularExpression("ab??cd"));
678   ASSERT_EQ("a.*b.c.*d", Toolbox::WildcardToRegularExpression("a*b?c*d"));
679   ASSERT_EQ("a\\{b\\]", Toolbox::WildcardToRegularExpression("a{b]"));
680 }
681 
682 
TEST(Toolbox,Tokenize)683 TEST(Toolbox, Tokenize)
684 {
685   std::vector<std::string> t;
686 
687   Toolbox::TokenizeString(t, "", ',');
688   ASSERT_EQ(1u, t.size());
689   ASSERT_EQ("", t[0]);
690 
691   Toolbox::TokenizeString(t, "abc", ',');
692   ASSERT_EQ(1u, t.size());
693   ASSERT_EQ("abc", t[0]);
694 
695   Toolbox::TokenizeString(t, "ab,cd,ef,", ',');
696   ASSERT_EQ(4u, t.size());
697   ASSERT_EQ("ab", t[0]);
698   ASSERT_EQ("cd", t[1]);
699   ASSERT_EQ("ef", t[2]);
700   ASSERT_EQ("", t[3]);
701 }
702 
TEST(Toolbox,Enumerations)703 TEST(Toolbox, Enumerations)
704 {
705   ASSERT_EQ(Encoding_Utf8, StringToEncoding(EnumerationToString(Encoding_Utf8)));
706   ASSERT_EQ(Encoding_Ascii, StringToEncoding(EnumerationToString(Encoding_Ascii)));
707   ASSERT_EQ(Encoding_Latin1, StringToEncoding(EnumerationToString(Encoding_Latin1)));
708   ASSERT_EQ(Encoding_Latin2, StringToEncoding(EnumerationToString(Encoding_Latin2)));
709   ASSERT_EQ(Encoding_Latin3, StringToEncoding(EnumerationToString(Encoding_Latin3)));
710   ASSERT_EQ(Encoding_Latin4, StringToEncoding(EnumerationToString(Encoding_Latin4)));
711   ASSERT_EQ(Encoding_Latin5, StringToEncoding(EnumerationToString(Encoding_Latin5)));
712   ASSERT_EQ(Encoding_Cyrillic, StringToEncoding(EnumerationToString(Encoding_Cyrillic)));
713   ASSERT_EQ(Encoding_Arabic, StringToEncoding(EnumerationToString(Encoding_Arabic)));
714   ASSERT_EQ(Encoding_Greek, StringToEncoding(EnumerationToString(Encoding_Greek)));
715   ASSERT_EQ(Encoding_Hebrew, StringToEncoding(EnumerationToString(Encoding_Hebrew)));
716   ASSERT_EQ(Encoding_Japanese, StringToEncoding(EnumerationToString(Encoding_Japanese)));
717   ASSERT_EQ(Encoding_Chinese, StringToEncoding(EnumerationToString(Encoding_Chinese)));
718   ASSERT_EQ(Encoding_Thai, StringToEncoding(EnumerationToString(Encoding_Thai)));
719   ASSERT_EQ(Encoding_Korean, StringToEncoding(EnumerationToString(Encoding_Korean)));
720   ASSERT_EQ(Encoding_JapaneseKanji, StringToEncoding(EnumerationToString(Encoding_JapaneseKanji)));
721   ASSERT_EQ(Encoding_SimplifiedChinese, StringToEncoding(EnumerationToString(Encoding_SimplifiedChinese)));
722 
723   ASSERT_EQ(ResourceType_Patient, StringToResourceType(EnumerationToString(ResourceType_Patient)));
724   ASSERT_EQ(ResourceType_Study, StringToResourceType(EnumerationToString(ResourceType_Study)));
725   ASSERT_EQ(ResourceType_Series, StringToResourceType(EnumerationToString(ResourceType_Series)));
726   ASSERT_EQ(ResourceType_Instance, StringToResourceType(EnumerationToString(ResourceType_Instance)));
727 
728   ASSERT_EQ(ImageFormat_Png, StringToImageFormat(EnumerationToString(ImageFormat_Png)));
729 
730   ASSERT_EQ(PhotometricInterpretation_ARGB, StringToPhotometricInterpretation(EnumerationToString(PhotometricInterpretation_ARGB)));
731   ASSERT_EQ(PhotometricInterpretation_CMYK, StringToPhotometricInterpretation(EnumerationToString(PhotometricInterpretation_CMYK)));
732   ASSERT_EQ(PhotometricInterpretation_HSV, StringToPhotometricInterpretation(EnumerationToString(PhotometricInterpretation_HSV)));
733   ASSERT_EQ(PhotometricInterpretation_Monochrome1, StringToPhotometricInterpretation(EnumerationToString(PhotometricInterpretation_Monochrome1)));
734   ASSERT_EQ(PhotometricInterpretation_Monochrome2, StringToPhotometricInterpretation(EnumerationToString(PhotometricInterpretation_Monochrome2)));
735   ASSERT_EQ(PhotometricInterpretation_Palette, StringToPhotometricInterpretation(EnumerationToString(PhotometricInterpretation_Palette)));
736   ASSERT_EQ(PhotometricInterpretation_RGB, StringToPhotometricInterpretation(EnumerationToString(PhotometricInterpretation_RGB)));
737   ASSERT_EQ(PhotometricInterpretation_YBRFull, StringToPhotometricInterpretation(EnumerationToString(PhotometricInterpretation_YBRFull)));
738   ASSERT_EQ(PhotometricInterpretation_YBRFull422, StringToPhotometricInterpretation(EnumerationToString(PhotometricInterpretation_YBRFull422)));
739   ASSERT_EQ(PhotometricInterpretation_YBRPartial420, StringToPhotometricInterpretation(EnumerationToString(PhotometricInterpretation_YBRPartial420)));
740   ASSERT_EQ(PhotometricInterpretation_YBRPartial422, StringToPhotometricInterpretation(EnumerationToString(PhotometricInterpretation_YBRPartial422)));
741   ASSERT_EQ(PhotometricInterpretation_YBR_ICT, StringToPhotometricInterpretation(EnumerationToString(PhotometricInterpretation_YBR_ICT)));
742   ASSERT_EQ(PhotometricInterpretation_YBR_RCT, StringToPhotometricInterpretation(EnumerationToString(PhotometricInterpretation_YBR_RCT)));
743 
744   ASSERT_STREQ("Unknown", EnumerationToString(PhotometricInterpretation_Unknown));
745   ASSERT_THROW(StringToPhotometricInterpretation("Unknown"), OrthancException);
746 
747   ASSERT_EQ(DicomVersion_2008, StringToDicomVersion(EnumerationToString(DicomVersion_2008)));
748   ASSERT_EQ(DicomVersion_2017c, StringToDicomVersion(EnumerationToString(DicomVersion_2017c)));
749 
750   for (int i = static_cast<int>(ValueRepresentation_ApplicationEntity);
751        i < static_cast<int>(ValueRepresentation_NotSupported); i += 1)
752   {
753     ValueRepresentation vr = static_cast<ValueRepresentation>(i);
754     ASSERT_EQ(vr, StringToValueRepresentation(EnumerationToString(vr), true));
755   }
756 
757   ASSERT_THROW(StringToValueRepresentation("nope", true), OrthancException);
758 
759   ASSERT_EQ(JobState_Pending, StringToJobState(EnumerationToString(JobState_Pending)));
760   ASSERT_EQ(JobState_Running, StringToJobState(EnumerationToString(JobState_Running)));
761   ASSERT_EQ(JobState_Success, StringToJobState(EnumerationToString(JobState_Success)));
762   ASSERT_EQ(JobState_Failure, StringToJobState(EnumerationToString(JobState_Failure)));
763   ASSERT_EQ(JobState_Paused, StringToJobState(EnumerationToString(JobState_Paused)));
764   ASSERT_EQ(JobState_Retry, StringToJobState(EnumerationToString(JobState_Retry)));
765   ASSERT_THROW(StringToJobState("nope"), OrthancException);
766 
767   ASSERT_EQ(MimeType_Binary, StringToMimeType(EnumerationToString(MimeType_Binary)));
768   ASSERT_EQ(MimeType_Css, StringToMimeType(EnumerationToString(MimeType_Css)));
769   ASSERT_EQ(MimeType_Dicom, StringToMimeType(EnumerationToString(MimeType_Dicom)));
770   ASSERT_EQ(MimeType_Gif, StringToMimeType(EnumerationToString(MimeType_Gif)));
771   ASSERT_EQ(MimeType_Gzip, StringToMimeType(EnumerationToString(MimeType_Gzip)));
772   ASSERT_EQ(MimeType_Html, StringToMimeType(EnumerationToString(MimeType_Html)));
773   ASSERT_EQ(MimeType_JavaScript, StringToMimeType(EnumerationToString(MimeType_JavaScript)));
774   ASSERT_EQ(MimeType_Jpeg, StringToMimeType(EnumerationToString(MimeType_Jpeg)));
775   ASSERT_EQ(MimeType_Jpeg2000, StringToMimeType(EnumerationToString(MimeType_Jpeg2000)));
776   ASSERT_EQ(MimeType_Json, StringToMimeType(EnumerationToString(MimeType_Json)));
777   ASSERT_EQ(MimeType_NaCl, StringToMimeType(EnumerationToString(MimeType_NaCl)));
778   ASSERT_EQ(MimeType_PNaCl, StringToMimeType(EnumerationToString(MimeType_PNaCl)));
779   ASSERT_EQ(MimeType_Pam, StringToMimeType(EnumerationToString(MimeType_Pam)));
780   ASSERT_EQ(MimeType_Pdf, StringToMimeType(EnumerationToString(MimeType_Pdf)));
781   ASSERT_EQ(MimeType_PlainText, StringToMimeType(EnumerationToString(MimeType_PlainText)));
782   ASSERT_EQ(MimeType_Png, StringToMimeType(EnumerationToString(MimeType_Png)));
783   ASSERT_EQ(MimeType_Svg, StringToMimeType(EnumerationToString(MimeType_Svg)));
784   ASSERT_EQ(MimeType_WebAssembly, StringToMimeType(EnumerationToString(MimeType_WebAssembly)));
785   ASSERT_EQ(MimeType_Xml, StringToMimeType("application/xml"));
786   ASSERT_EQ(MimeType_Xml, StringToMimeType("text/xml"));
787   ASSERT_EQ(MimeType_Xml, StringToMimeType(EnumerationToString(MimeType_Xml)));
788   ASSERT_EQ(MimeType_DicomWebJson, StringToMimeType(EnumerationToString(MimeType_DicomWebJson)));
789   ASSERT_EQ(MimeType_DicomWebXml, StringToMimeType(EnumerationToString(MimeType_DicomWebXml)));
790   ASSERT_THROW(StringToMimeType("nope"), OrthancException);
791 
792   ASSERT_TRUE(IsResourceLevelAboveOrEqual(ResourceType_Patient, ResourceType_Patient));
793   ASSERT_TRUE(IsResourceLevelAboveOrEqual(ResourceType_Patient, ResourceType_Study));
794   ASSERT_TRUE(IsResourceLevelAboveOrEqual(ResourceType_Patient, ResourceType_Series));
795   ASSERT_TRUE(IsResourceLevelAboveOrEqual(ResourceType_Patient, ResourceType_Instance));
796 
797   ASSERT_FALSE(IsResourceLevelAboveOrEqual(ResourceType_Study, ResourceType_Patient));
798   ASSERT_TRUE(IsResourceLevelAboveOrEqual(ResourceType_Study, ResourceType_Study));
799   ASSERT_TRUE(IsResourceLevelAboveOrEqual(ResourceType_Study, ResourceType_Series));
800   ASSERT_TRUE(IsResourceLevelAboveOrEqual(ResourceType_Study, ResourceType_Instance));
801 
802   ASSERT_FALSE(IsResourceLevelAboveOrEqual(ResourceType_Series, ResourceType_Patient));
803   ASSERT_FALSE(IsResourceLevelAboveOrEqual(ResourceType_Series, ResourceType_Study));
804   ASSERT_TRUE(IsResourceLevelAboveOrEqual(ResourceType_Series, ResourceType_Series));
805   ASSERT_TRUE(IsResourceLevelAboveOrEqual(ResourceType_Series, ResourceType_Instance));
806 
807   ASSERT_FALSE(IsResourceLevelAboveOrEqual(ResourceType_Instance, ResourceType_Patient));
808   ASSERT_FALSE(IsResourceLevelAboveOrEqual(ResourceType_Instance, ResourceType_Study));
809   ASSERT_FALSE(IsResourceLevelAboveOrEqual(ResourceType_Instance, ResourceType_Series));
810   ASSERT_TRUE(IsResourceLevelAboveOrEqual(ResourceType_Instance, ResourceType_Instance));
811 
812   ASSERT_STREQ("Patients", GetResourceTypeText(ResourceType_Patient, true /* plural */, true /* upper case */));
813   ASSERT_STREQ("patients", GetResourceTypeText(ResourceType_Patient, true, false));
814   ASSERT_STREQ("Patient", GetResourceTypeText(ResourceType_Patient, false, true));
815   ASSERT_STREQ("patient", GetResourceTypeText(ResourceType_Patient, false, false));
816   ASSERT_STREQ("Studies", GetResourceTypeText(ResourceType_Study, true, true));
817   ASSERT_STREQ("studies", GetResourceTypeText(ResourceType_Study, true, false));
818   ASSERT_STREQ("Study", GetResourceTypeText(ResourceType_Study, false, true));
819   ASSERT_STREQ("study", GetResourceTypeText(ResourceType_Study, false, false));
820   ASSERT_STREQ("Series", GetResourceTypeText(ResourceType_Series, true, true));
821   ASSERT_STREQ("series", GetResourceTypeText(ResourceType_Series, true, false));
822   ASSERT_STREQ("Series", GetResourceTypeText(ResourceType_Series, false, true));
823   ASSERT_STREQ("series", GetResourceTypeText(ResourceType_Series, false, false));
824   ASSERT_STREQ("Instances", GetResourceTypeText(ResourceType_Instance, true, true));
825   ASSERT_STREQ("instances", GetResourceTypeText(ResourceType_Instance, true, false));
826   ASSERT_STREQ("Instance", GetResourceTypeText(ResourceType_Instance, false, true));
827   ASSERT_STREQ("instance", GetResourceTypeText(ResourceType_Instance, false, false));
828 }
829 
830 
831 #if defined(__linux__) || defined(__OpenBSD__)
832 #include <endian.h>
833 #elif defined(__FreeBSD__)
834 #include <machine/endian.h>
835 #endif
836 
837 
TEST(Toolbox,Endianness)838 TEST(Toolbox, Endianness)
839 {
840   // Parts of this test come from Adam Conrad
841   // http://bugs.debian.org/cgi-bin/bugreport.cgi?bug=728822#5
842 
843 
844   /**
845    * Windows and OS X are assumed to always little-endian.
846    **/
847 
848 #if defined(_WIN32) || defined(__APPLE__)
849   ASSERT_EQ(Endianness_Little, Toolbox::DetectEndianness());
850 
851 
852   /**
853    * FreeBSD.
854    **/
855 
856 #elif defined(__FreeBSD__) || defined(__OpenBSD__)
857 #  if _BYTE_ORDER == _BIG_ENDIAN
858    ASSERT_EQ(Endianness_Big, Toolbox::DetectEndianness());
859 #  else // _LITTLE_ENDIAN
860    ASSERT_EQ(Endianness_Little, Toolbox::DetectEndianness());
861 #  endif
862 
863 
864   /**
865    * Linux.
866    **/
867 
868 #elif defined(__linux__) || defined(__FreeBSD_kernel__)
869 
870 #if !defined(__BYTE_ORDER)
871 #  error Support your platform here
872 #endif
873 
874 #  if __BYTE_ORDER == __BIG_ENDIAN
875   ASSERT_EQ(Endianness_Big, Toolbox::DetectEndianness());
876 #  else // __LITTLE_ENDIAN
877   ASSERT_EQ(Endianness_Little, Toolbox::DetectEndianness());
878 #  endif
879 
880 
881   /**
882    * WebAssembly is always little-endian.
883    **/
884 
885 #elif defined(__EMSCRIPTEN__)
886   ASSERT_EQ(Endianness_Little, Toolbox::DetectEndianness());
887 #else
888 #  error Support your platform here
889 #endif
890 }
891 
892 
893 #include "../Sources/Endianness.h"
894 
ASSERT_EQ16(uint16_t a,uint16_t b)895 static void ASSERT_EQ16(uint16_t a, uint16_t b)
896 {
897 #ifdef __MINGW32__
898   // This cast solves a linking problem with MinGW
899   ASSERT_EQ(static_cast<unsigned int>(a), static_cast<unsigned int>(b));
900 #else
901   ASSERT_EQ(a, b);
902 #endif
903 }
904 
ASSERT_NE16(uint16_t a,uint16_t b)905 static void ASSERT_NE16(uint16_t a, uint16_t b)
906 {
907 #ifdef __MINGW32__
908   // This cast solves a linking problem with MinGW
909   ASSERT_NE(static_cast<unsigned int>(a), static_cast<unsigned int>(b));
910 #else
911   ASSERT_NE(a, b);
912 #endif
913 }
914 
ASSERT_EQ32(uint32_t a,uint32_t b)915 static void ASSERT_EQ32(uint32_t a, uint32_t b)
916 {
917 #ifdef __MINGW32__
918   // This cast solves a linking problem with MinGW
919   ASSERT_EQ(static_cast<unsigned int>(a), static_cast<unsigned int>(b));
920 #else
921   ASSERT_EQ(a, b);
922 #endif
923 }
924 
ASSERT_NE32(uint32_t a,uint32_t b)925 static void ASSERT_NE32(uint32_t a, uint32_t b)
926 {
927 #ifdef __MINGW32__
928   // This cast solves a linking problem with MinGW
929   ASSERT_NE(static_cast<unsigned int>(a), static_cast<unsigned int>(b));
930 #else
931   ASSERT_NE(a, b);
932 #endif
933 }
934 
ASSERT_EQ64(uint64_t a,uint64_t b)935 static void ASSERT_EQ64(uint64_t a, uint64_t b)
936 {
937 #ifdef __MINGW32__
938   // This cast solves a linking problem with MinGW
939   ASSERT_EQ(static_cast<unsigned int>(a), static_cast<unsigned int>(b));
940 #else
941   ASSERT_EQ(a, b);
942 #endif
943 }
944 
ASSERT_NE64(uint64_t a,uint64_t b)945 static void ASSERT_NE64(uint64_t a, uint64_t b)
946 {
947 #ifdef __MINGW32__
948   // This cast solves a linking problem with MinGW
949   ASSERT_NE(static_cast<unsigned long long>(a), static_cast<unsigned long long>(b));
950 #else
951   ASSERT_NE(a, b);
952 #endif
953 }
954 
955 
956 
TEST(Toolbox,EndiannessConversions16)957 TEST(Toolbox, EndiannessConversions16)
958 {
959   Endianness e = Toolbox::DetectEndianness();
960 
961   for (unsigned int i = 0; i < 65536; i += 17)
962   {
963     uint16_t v = static_cast<uint16_t>(i);
964     ASSERT_EQ16(v, be16toh(htobe16(v)));
965     ASSERT_EQ16(v, le16toh(htole16(v)));
966 
967     const uint8_t* bytes = reinterpret_cast<const uint8_t*>(&v);
968     if (bytes[0] != bytes[1])
969     {
970       ASSERT_NE16(v, le16toh(htobe16(v)));
971       ASSERT_NE16(v, be16toh(htole16(v)));
972     }
973     else
974     {
975       ASSERT_EQ16(v, le16toh(htobe16(v)));
976       ASSERT_EQ16(v, be16toh(htole16(v)));
977     }
978 
979     switch (e)
980     {
981       case Endianness_Little:
982         ASSERT_EQ16(v, htole16(v));
983         if (bytes[0] != bytes[1])
984         {
985           ASSERT_NE16(v, htobe16(v));
986         }
987         else
988         {
989           ASSERT_EQ16(v, htobe16(v));
990         }
991         break;
992 
993       case Endianness_Big:
994         ASSERT_EQ16(v, htobe16(v));
995         if (bytes[0] != bytes[1])
996         {
997           ASSERT_NE16(v, htole16(v));
998         }
999         else
1000         {
1001           ASSERT_EQ16(v, htole16(v));
1002         }
1003         break;
1004 
1005       default:
1006         throw OrthancException(ErrorCode_ParameterOutOfRange);
1007     }
1008   }
1009 }
1010 
1011 
TEST(Toolbox,EndiannessConversions32)1012 TEST(Toolbox, EndiannessConversions32)
1013 {
1014   const uint32_t v = 0xff010203u;
1015   const uint32_t r = 0x030201ffu;
1016   ASSERT_EQ32(v, be32toh(htobe32(v)));
1017   ASSERT_EQ32(v, le32toh(htole32(v)));
1018   ASSERT_NE32(v, be32toh(htole32(v)));
1019   ASSERT_NE32(v, le32toh(htobe32(v)));
1020 
1021   switch (Toolbox::DetectEndianness())
1022   {
1023     case Endianness_Little:
1024       ASSERT_EQ32(r, htobe32(v));
1025       ASSERT_EQ32(v, htole32(v));
1026       ASSERT_EQ32(r, be32toh(v));
1027       ASSERT_EQ32(v, le32toh(v));
1028       break;
1029 
1030     case Endianness_Big:
1031       ASSERT_EQ32(v, htobe32(v));
1032       ASSERT_EQ32(r, htole32(v));
1033       ASSERT_EQ32(v, be32toh(v));
1034       ASSERT_EQ32(r, le32toh(v));
1035       break;
1036 
1037     default:
1038       throw OrthancException(ErrorCode_ParameterOutOfRange);
1039   }
1040 }
1041 
1042 
TEST(Toolbox,EndiannessConversions64)1043 TEST(Toolbox, EndiannessConversions64)
1044 {
1045   const uint64_t v = 0xff01020304050607LL;
1046   const uint64_t r = 0x07060504030201ffLL;
1047   ASSERT_EQ64(v, be64toh(htobe64(v)));
1048   ASSERT_EQ64(v, le64toh(htole64(v)));
1049   ASSERT_NE64(v, be64toh(htole64(v)));
1050   ASSERT_NE64(v, le64toh(htobe64(v)));
1051 
1052   switch (Toolbox::DetectEndianness())
1053   {
1054     case Endianness_Little:
1055       ASSERT_EQ64(r, htobe64(v));
1056       ASSERT_EQ64(v, htole64(v));
1057       ASSERT_EQ64(r, be64toh(v));
1058       ASSERT_EQ64(v, le64toh(v));
1059       break;
1060 
1061     case Endianness_Big:
1062       ASSERT_EQ64(v, htobe64(v));
1063       ASSERT_EQ64(r, htole64(v));
1064       ASSERT_EQ64(v, be64toh(v));
1065       ASSERT_EQ64(r, le64toh(v));
1066       break;
1067 
1068     default:
1069       throw OrthancException(ErrorCode_ParameterOutOfRange);
1070   }
1071 }
1072 
1073 
1074 #if ORTHANC_SANDBOXED != 1
TEST(Toolbox,Now)1075 TEST(Toolbox, Now)
1076 {
1077   LOG(WARNING) << "Local time: " << SystemToolbox::GetNowIsoString(false);
1078   LOG(WARNING) << "Universal time: " << SystemToolbox::GetNowIsoString(true);
1079 
1080   std::string date, time;
1081   SystemToolbox::GetNowDicom(date, time, false);
1082   LOG(WARNING) << "Local DICOM time: [" << date << "] [" << time << "]";
1083 
1084   SystemToolbox::GetNowDicom(date, time, true);
1085   LOG(WARNING) << "Universal DICOM time: [" << date << "] [" << time << "]";
1086 }
1087 #endif
1088 
1089 
1090 #if ORTHANC_ENABLE_PUGIXML == 1
TEST(Toolbox,Xml)1091 TEST(Toolbox, Xml)
1092 {
1093   Json::Value a;
1094   a["hello"] = "world";
1095   a["42"] = 43;
1096   a["b"] = Json::arrayValue;
1097   a["b"].append("test");
1098   a["b"].append("test2");
1099 
1100   std::string s;
1101   Toolbox::JsonToXml(s, a);
1102 
1103   std::cout << s;
1104 }
1105 #endif
1106 
1107 
1108 #if !defined(_WIN32) && (ORTHANC_SANDBOXED != 1)
TEST(Toolbox,ExecuteSystemCommand)1109 TEST(Toolbox, ExecuteSystemCommand)
1110 {
1111   std::vector<std::string> args(2);
1112   args[0] = "Hello";
1113   args[1] = "World";
1114 
1115   SystemToolbox::ExecuteSystemCommand("echo", args);
1116 }
1117 #endif
1118 
1119 
TEST(Toolbox,IsInteger)1120 TEST(Toolbox, IsInteger)
1121 {
1122   ASSERT_TRUE(Toolbox::IsInteger("00236"));
1123   ASSERT_TRUE(Toolbox::IsInteger("-0042"));
1124   ASSERT_TRUE(Toolbox::IsInteger("0"));
1125   ASSERT_TRUE(Toolbox::IsInteger("-0"));
1126 
1127   ASSERT_FALSE(Toolbox::IsInteger(""));
1128   ASSERT_FALSE(Toolbox::IsInteger("42a"));
1129   ASSERT_FALSE(Toolbox::IsInteger("42-"));
1130 }
1131 
1132 
TEST(Toolbox,StartsWith)1133 TEST(Toolbox, StartsWith)
1134 {
1135   ASSERT_TRUE(Toolbox::StartsWith("hello world", ""));
1136   ASSERT_TRUE(Toolbox::StartsWith("hello world", "hello"));
1137   ASSERT_TRUE(Toolbox::StartsWith("hello world", "h"));
1138   ASSERT_FALSE(Toolbox::StartsWith("hello world", "H"));
1139   ASSERT_FALSE(Toolbox::StartsWith("h", "hello"));
1140   ASSERT_TRUE(Toolbox::StartsWith("h", "h"));
1141   ASSERT_FALSE(Toolbox::StartsWith("", "h"));
1142 }
1143 
1144 
TEST(Toolbox,UriEncode)1145 TEST(Toolbox, UriEncode)
1146 {
1147   std::string s;
1148 
1149   // Unreserved characters must not be modified
1150   std::string t = "aAzZ09.-~_";
1151   Toolbox::UriEncode(s, t);
1152   ASSERT_EQ(t, s);
1153 
1154   Toolbox::UriEncode(s, "!#$&'()*+,/:;=?@[]"); ASSERT_EQ("%21%23%24%26%27%28%29%2A%2B%2C/%3A%3B%3D%3F%40%5B%5D", s);
1155   Toolbox::UriEncode(s, "%"); ASSERT_EQ("%25", s);
1156 
1157   // Encode characters from UTF-8. This is the test string from the
1158   // file "../Resources/EncodingTests.py"
1159   Toolbox::UriEncode(s, "\x54\x65\x73\x74\xc3\xa9\xc3\xa4\xc3\xb6\xc3\xb2\xd0\x94\xce\x98\xc4\x9d\xd7\x93\xd8\xb5\xc4\xb7\xd1\x9b\xe0\xb9\x9b\xef\xbe\x88\xc4\xb0");
1160   ASSERT_EQ("Test%C3%A9%C3%A4%C3%B6%C3%B2%D0%94%CE%98%C4%9D%D7%93%D8%B5%C4%B7%D1%9B%E0%B9%9B%EF%BE%88%C4%B0", s);
1161 }
1162 
1163 
TEST(Toolbox,AccessJson)1164 TEST(Toolbox, AccessJson)
1165 {
1166   Json::Value v = Json::arrayValue;
1167   ASSERT_EQ("nope", Toolbox::GetJsonStringField(v, "hello", "nope"));
1168 
1169   v = Json::objectValue;
1170   ASSERT_EQ("nope", Toolbox::GetJsonStringField(v, "hello", "nope"));
1171   ASSERT_EQ(-10, Toolbox::GetJsonIntegerField(v, "hello", -10));
1172   ASSERT_EQ(10u, Toolbox::GetJsonUnsignedIntegerField(v, "hello", 10));
1173   ASSERT_TRUE(Toolbox::GetJsonBooleanField(v, "hello", true));
1174 
1175   v["hello"] = "world";
1176   ASSERT_EQ("world", Toolbox::GetJsonStringField(v, "hello", "nope"));
1177   ASSERT_THROW(Toolbox::GetJsonIntegerField(v, "hello", -10), OrthancException);
1178   ASSERT_THROW(Toolbox::GetJsonUnsignedIntegerField(v, "hello", 10), OrthancException);
1179   ASSERT_THROW(Toolbox::GetJsonBooleanField(v, "hello", true), OrthancException);
1180 
1181   v["hello"] = -42;
1182   ASSERT_THROW(Toolbox::GetJsonStringField(v, "hello", "nope"), OrthancException);
1183   ASSERT_EQ(-42, Toolbox::GetJsonIntegerField(v, "hello", -10));
1184   ASSERT_THROW(Toolbox::GetJsonUnsignedIntegerField(v, "hello", 10), OrthancException);
1185   ASSERT_THROW(Toolbox::GetJsonBooleanField(v, "hello", true), OrthancException);
1186 
1187   v["hello"] = 42;
1188   ASSERT_THROW(Toolbox::GetJsonStringField(v, "hello", "nope"), OrthancException);
1189   ASSERT_EQ(42, Toolbox::GetJsonIntegerField(v, "hello", -10));
1190   ASSERT_EQ(42u, Toolbox::GetJsonUnsignedIntegerField(v, "hello", 10));
1191   ASSERT_THROW(Toolbox::GetJsonBooleanField(v, "hello", true), OrthancException);
1192 
1193   v["hello"] = false;
1194   ASSERT_THROW(Toolbox::GetJsonStringField(v, "hello", "nope"), OrthancException);
1195   ASSERT_THROW(Toolbox::GetJsonIntegerField(v, "hello", -10), OrthancException);
1196   ASSERT_THROW(Toolbox::GetJsonUnsignedIntegerField(v, "hello", 10), OrthancException);
1197   ASSERT_FALSE(Toolbox::GetJsonBooleanField(v, "hello", true));
1198 }
1199 
1200 
TEST(Toolbox,LinesIterator)1201 TEST(Toolbox, LinesIterator)
1202 {
1203   std::string s;
1204 
1205   {
1206     std::string content;
1207     Toolbox::LinesIterator it(content);
1208     ASSERT_FALSE(it.GetLine(s));
1209   }
1210 
1211   {
1212     std::string content = "\n\r";
1213     Toolbox::LinesIterator it(content);
1214     ASSERT_TRUE(it.GetLine(s)); it.Next(); ASSERT_EQ("", s);
1215     ASSERT_FALSE(it.GetLine(s));
1216   }
1217 
1218   {
1219     std::string content = "\n Hello \n\nWorld\n\n";
1220     Toolbox::LinesIterator it(content);
1221     ASSERT_TRUE(it.GetLine(s)); it.Next(); ASSERT_EQ("", s);
1222     ASSERT_TRUE(it.GetLine(s)); it.Next(); ASSERT_EQ(" Hello ", s);
1223     ASSERT_TRUE(it.GetLine(s)); it.Next(); ASSERT_EQ("", s);
1224     ASSERT_TRUE(it.GetLine(s)); it.Next(); ASSERT_EQ("World", s);
1225     ASSERT_TRUE(it.GetLine(s)); it.Next(); ASSERT_EQ("", s);
1226     ASSERT_FALSE(it.GetLine(s)); it.Next();
1227     ASSERT_FALSE(it.GetLine(s));
1228   }
1229 
1230   {
1231     std::string content = "\r Hello \r\rWorld\r\r";
1232     Toolbox::LinesIterator it(content);
1233     ASSERT_TRUE(it.GetLine(s)); it.Next(); ASSERT_EQ("", s);
1234     ASSERT_TRUE(it.GetLine(s)); it.Next(); ASSERT_EQ(" Hello ", s);
1235     ASSERT_TRUE(it.GetLine(s)); it.Next(); ASSERT_EQ("", s);
1236     ASSERT_TRUE(it.GetLine(s)); it.Next(); ASSERT_EQ("World", s);
1237     ASSERT_TRUE(it.GetLine(s)); it.Next(); ASSERT_EQ("", s);
1238     ASSERT_FALSE(it.GetLine(s)); it.Next();
1239     ASSERT_FALSE(it.GetLine(s));
1240   }
1241 
1242   {
1243     std::string content = "\n\r Hello \n\r\n\rWorld\n\r\n\r";
1244     Toolbox::LinesIterator it(content);
1245     ASSERT_TRUE(it.GetLine(s)); it.Next(); ASSERT_EQ("", s);
1246     ASSERT_TRUE(it.GetLine(s)); it.Next(); ASSERT_EQ(" Hello ", s);
1247     ASSERT_TRUE(it.GetLine(s)); it.Next(); ASSERT_EQ("", s);
1248     ASSERT_TRUE(it.GetLine(s)); it.Next(); ASSERT_EQ("World", s);
1249     ASSERT_TRUE(it.GetLine(s)); it.Next(); ASSERT_EQ("", s);
1250     ASSERT_FALSE(it.GetLine(s)); it.Next();
1251     ASSERT_FALSE(it.GetLine(s));
1252   }
1253 
1254   {
1255     std::string content = "\r\n Hello \r\n\r\nWorld\r\n\r\n";
1256     Toolbox::LinesIterator it(content);
1257     ASSERT_TRUE(it.GetLine(s)); it.Next(); ASSERT_EQ("", s);
1258     ASSERT_TRUE(it.GetLine(s)); it.Next(); ASSERT_EQ(" Hello ", s);
1259     ASSERT_TRUE(it.GetLine(s)); it.Next(); ASSERT_EQ("", s);
1260     ASSERT_TRUE(it.GetLine(s)); it.Next(); ASSERT_EQ("World", s);
1261     ASSERT_TRUE(it.GetLine(s)); it.Next(); ASSERT_EQ("", s);
1262     ASSERT_FALSE(it.GetLine(s)); it.Next();
1263     ASSERT_FALSE(it.GetLine(s));
1264   }
1265 }
1266 
1267 
1268 #if ORTHANC_SANDBOXED != 1
TEST(Toolbox,SubstituteVariables)1269 TEST(Toolbox, SubstituteVariables)
1270 {
1271   std::map<std::string, std::string> env;
1272   env["NOPE"] = "nope";
1273   env["WORLD"] = "world";
1274 
1275   ASSERT_EQ("Hello world\r\nWorld \r\nDone world\r\n",
1276             Toolbox::SubstituteVariables(
1277               "Hello ${WORLD}\r\nWorld ${HELLO}\r\nDone ${WORLD}\r\n",
1278               env));
1279 
1280   ASSERT_EQ("world A a B world C 'c' D {\"a\":\"b\"} E ",
1281             Toolbox::SubstituteVariables(
1282               "${WORLD} A ${WORLD2:-a} B ${WORLD:-b} C ${WORLD2:-\"'c'\"} D ${WORLD2:-'{\"a\":\"b\"}'} E ${WORLD2:-}",
1283               env));
1284 
1285   SystemToolbox::GetEnvironmentVariables(env);
1286   ASSERT_TRUE(env.find("NOPE") == env.end());
1287 
1288   // The "PATH" environment variable should always be available on
1289   // machines running the unit tests
1290   ASSERT_TRUE(env.find("PATH") != env.end() /* Case used by UNIX */ ||
1291               env.find("Path") != env.end() /* Case used by Windows */);
1292 
1293   env["PATH"] = "hello";
1294   ASSERT_EQ("AhelloB",
1295             Toolbox::SubstituteVariables("A${PATH}B", env));
1296 }
1297 #endif
1298 
1299 
1300 #if ORTHANC_SANDBOXED != 1
TEST(MetricsRegistry,Basic)1301 TEST(MetricsRegistry, Basic)
1302 {
1303   {
1304     MetricsRegistry m;
1305     m.SetEnabled(false);
1306     m.SetValue("hello.world", 42.5f);
1307 
1308     std::string s;
1309     m.ExportPrometheusText(s);
1310     ASSERT_TRUE(s.empty());
1311   }
1312 
1313   {
1314     MetricsRegistry m;
1315     m.Register("hello.world", MetricsType_Default);
1316 
1317     std::string s;
1318     m.ExportPrometheusText(s);
1319     ASSERT_TRUE(s.empty());
1320   }
1321 
1322   {
1323     MetricsRegistry m;
1324     m.SetValue("hello.world", 42.5f);
1325     ASSERT_EQ(MetricsType_Default, m.GetMetricsType("hello.world"));
1326     ASSERT_THROW(m.GetMetricsType("nope"), OrthancException);
1327 
1328     std::string s;
1329     m.ExportPrometheusText(s);
1330 
1331     std::vector<std::string> t;
1332     Toolbox::TokenizeString(t, s, '\n');
1333     ASSERT_EQ(2u, t.size());
1334     ASSERT_EQ("hello.world 42.5 ", t[0].substr(0, 17));
1335     ASSERT_TRUE(t[1].empty());
1336   }
1337 
1338   {
1339     MetricsRegistry m;
1340     m.Register("hello.max", MetricsType_MaxOver10Seconds);
1341     m.SetValue("hello.max", 10);
1342     m.SetValue("hello.max", 20);
1343     m.SetValue("hello.max", -10);
1344     m.SetValue("hello.max", 5);
1345 
1346     m.Register("hello.min", MetricsType_MinOver10Seconds);
1347     m.SetValue("hello.min", 10);
1348     m.SetValue("hello.min", 20);
1349     m.SetValue("hello.min", -10);
1350     m.SetValue("hello.min", 5);
1351 
1352     m.Register("hello.default", MetricsType_Default);
1353     m.SetValue("hello.default", 10);
1354     m.SetValue("hello.default", 20);
1355     m.SetValue("hello.default", -10);
1356     m.SetValue("hello.default", 5);
1357 
1358     ASSERT_EQ(MetricsType_MaxOver10Seconds, m.GetMetricsType("hello.max"));
1359     ASSERT_EQ(MetricsType_MinOver10Seconds, m.GetMetricsType("hello.min"));
1360     ASSERT_EQ(MetricsType_Default, m.GetMetricsType("hello.default"));
1361 
1362     std::string s;
1363     m.ExportPrometheusText(s);
1364 
1365     std::vector<std::string> t;
1366     Toolbox::TokenizeString(t, s, '\n');
1367     ASSERT_EQ(4u, t.size());
1368     ASSERT_TRUE(t[3].empty());
1369 
1370     std::map<std::string, std::string> u;
1371     for (size_t i = 0; i < t.size() - 1; i++)
1372     {
1373       std::vector<std::string> v;
1374       Toolbox::TokenizeString(v, t[i], ' ');
1375       u[v[0]] = v[1];
1376     }
1377 
1378     ASSERT_EQ("20", u["hello.max"]);
1379     ASSERT_EQ("-10", u["hello.min"]);
1380     ASSERT_EQ("5", u["hello.default"]);
1381   }
1382 
1383   {
1384     MetricsRegistry m;
1385 
1386     m.SetValue("a", 10);
1387     m.SetValue("b", 10, MetricsType_MinOver10Seconds);
1388 
1389     m.Register("c", MetricsType_MaxOver10Seconds);
1390     m.SetValue("c", 10, MetricsType_MinOver10Seconds);
1391 
1392     m.Register("d", MetricsType_MaxOver10Seconds);
1393     m.Register("d", MetricsType_Default);
1394 
1395     ASSERT_EQ(MetricsType_Default, m.GetMetricsType("a"));
1396     ASSERT_EQ(MetricsType_MinOver10Seconds, m.GetMetricsType("b"));
1397     ASSERT_EQ(MetricsType_MaxOver10Seconds, m.GetMetricsType("c"));
1398     ASSERT_EQ(MetricsType_Default, m.GetMetricsType("d"));
1399   }
1400 
1401   {
1402     MetricsRegistry m;
1403 
1404     {
1405       MetricsRegistry::Timer t1(m, "a");
1406       MetricsRegistry::Timer t2(m, "b", MetricsType_MinOver10Seconds);
1407     }
1408 
1409     ASSERT_EQ(MetricsType_MaxOver10Seconds, m.GetMetricsType("a"));
1410     ASSERT_EQ(MetricsType_MinOver10Seconds, m.GetMetricsType("b"));
1411   }
1412 }
1413 #endif
1414 
1415 
1416 #if ORTHANC_SANDBOXED != 1
TEST(Toolbox,ReadFileRange)1417 TEST(Toolbox, ReadFileRange)
1418 {
1419   TemporaryFile tmp;
1420   std::string s;
1421 
1422   tmp.Write("");
1423   tmp.Read(s);                     ASSERT_TRUE(s.empty());
1424   tmp.ReadRange(s, 0, 0, true);    ASSERT_TRUE(s.empty());
1425   tmp.ReadRange(s, 0, 10, false);  ASSERT_TRUE(s.empty());
1426 
1427   ASSERT_THROW(tmp.ReadRange(s, 0, 1, true), OrthancException);
1428 
1429   tmp.Write("Hello");
1430   tmp.Read(s);                     ASSERT_EQ("Hello", s);
1431   tmp.ReadRange(s, 0, 5, true);    ASSERT_EQ("Hello", s);
1432   tmp.ReadRange(s, 0, 1, true);    ASSERT_EQ("H", s);
1433   tmp.ReadRange(s, 1, 2, true);    ASSERT_EQ("e", s);
1434   tmp.ReadRange(s, 2, 3, true);    ASSERT_EQ("l", s);
1435   tmp.ReadRange(s, 3, 4, true);    ASSERT_EQ("l", s);
1436   tmp.ReadRange(s, 4, 5, true);    ASSERT_EQ("o", s);
1437   tmp.ReadRange(s, 2, 5, true);    ASSERT_EQ("llo", s);
1438   tmp.ReadRange(s, 2, 50, false);  ASSERT_EQ("llo", s);
1439   tmp.ReadRange(s, 2, 2, false);   ASSERT_TRUE(s.empty());
1440   tmp.ReadRange(s, 10, 50, false); ASSERT_TRUE(s.empty());
1441 
1442   ASSERT_THROW(tmp.ReadRange(s, 5, 10, true), OrthancException);
1443   ASSERT_THROW(tmp.ReadRange(s, 10, 50, true), OrthancException);
1444   ASSERT_THROW(tmp.ReadRange(s, 50, 10, true), OrthancException);
1445   ASSERT_THROW(tmp.ReadRange(s, 2, 1, true), OrthancException);
1446 }
1447 #endif
1448 
1449 
1450 #if ORTHANC_SANDBOXED != 1
TEST(Toolbox,GetMacAddressess)1451 TEST(Toolbox, GetMacAddressess)
1452 {
1453   std::set<std::string> mac;
1454   SystemToolbox::GetMacAddresses(mac);
1455 
1456   for (std::set<std::string>::const_iterator it = mac.begin(); it != mac.end(); ++it)
1457   {
1458     printf("MAC address: [%s]\n", it->c_str());
1459   }
1460 }
1461 #endif
1462