UtilTest.cc 29 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922
  1. #include "util.h"
  2. #include <cstring>
  3. #include <string>
  4. #include <iostream>
  5. #include <cppunit/extensions/HelperMacros.h>
  6. #include "FixedNumberRandomizer.h"
  7. #include "DlAbortEx.h"
  8. #include "BitfieldMan.h"
  9. #include "ByteArrayDiskWriter.h"
  10. #include "FileEntry.h"
  11. #include "File.h"
  12. #include "array_fun.h"
  13. namespace aria2 {
  14. class UtilTest:public CppUnit::TestFixture {
  15. CPPUNIT_TEST_SUITE(UtilTest);
  16. CPPUNIT_TEST(testTrim);
  17. CPPUNIT_TEST(testSplit);
  18. CPPUNIT_TEST(testSplit_many);
  19. CPPUNIT_TEST(testEndsWith);
  20. CPPUNIT_TEST(testReplace);
  21. CPPUNIT_TEST(testStartsWith);
  22. // may be moved to other helper class in the future.
  23. CPPUNIT_TEST(testGetContentDispositionFilename);
  24. CPPUNIT_TEST(testRandomAlpha);
  25. CPPUNIT_TEST(testToUpper);
  26. CPPUNIT_TEST(testToLower);
  27. CPPUNIT_TEST(testUrldecode);
  28. CPPUNIT_TEST(testGetRealSize);
  29. CPPUNIT_TEST(testAbbrevSize);
  30. CPPUNIT_TEST(testToStream);
  31. CPPUNIT_TEST(testIsNumber);
  32. CPPUNIT_TEST(testIsLowercase);
  33. CPPUNIT_TEST(testIsUppercase);
  34. CPPUNIT_TEST(testAlphaToNum);
  35. CPPUNIT_TEST(testMkdirs);
  36. CPPUNIT_TEST(testConvertBitfield);
  37. CPPUNIT_TEST(testParseIntRange);
  38. CPPUNIT_TEST(testParseIntRange_invalidRange);
  39. CPPUNIT_TEST(testParseInt);
  40. CPPUNIT_TEST(testParseUInt);
  41. CPPUNIT_TEST(testParseLLInt);
  42. CPPUNIT_TEST(testParseULLInt);
  43. CPPUNIT_TEST(testToString_binaryStream);
  44. CPPUNIT_TEST(testItos);
  45. CPPUNIT_TEST(testUitos);
  46. CPPUNIT_TEST(testNtoh64);
  47. CPPUNIT_TEST(testUrlencode);
  48. CPPUNIT_TEST(testHtmlEscape);
  49. CPPUNIT_TEST(testJoinPath);
  50. CPPUNIT_TEST(testParseIndexPath);
  51. CPPUNIT_TEST(testCreateIndexPathMap);
  52. CPPUNIT_TEST(testGenerateRandomData);
  53. CPPUNIT_TEST(testFromHex);
  54. CPPUNIT_TEST(testParsePrioritizePieceRange);
  55. CPPUNIT_TEST(testApplyDir);
  56. CPPUNIT_TEST(testFixTaintedBasename);
  57. CPPUNIT_TEST_SUITE_END();
  58. private:
  59. public:
  60. void setUp() {
  61. }
  62. void testTrim();
  63. void testSplit();
  64. void testSplit_many();
  65. void testEndsWith();
  66. void testReplace();
  67. void testStartsWith();
  68. // may be moved to other helper class in the future.
  69. void testGetContentDispositionFilename();
  70. void testRandomAlpha();
  71. void testToUpper();
  72. void testToLower();
  73. void testUrldecode();
  74. void testGetRealSize();
  75. void testAbbrevSize();
  76. void testToStream();
  77. void testIsNumber();
  78. void testIsLowercase();
  79. void testIsUppercase();
  80. void testAlphaToNum();
  81. void testMkdirs();
  82. void testConvertBitfield();
  83. void testParseIntRange();
  84. void testParseIntRange_invalidRange();
  85. void testParseInt();
  86. void testParseUInt();
  87. void testParseLLInt();
  88. void testParseULLInt();
  89. void testToString_binaryStream();
  90. void testItos();
  91. void testUitos();
  92. void testNtoh64();
  93. void testUrlencode();
  94. void testHtmlEscape();
  95. void testJoinPath();
  96. void testParseIndexPath();
  97. void testCreateIndexPathMap();
  98. void testGenerateRandomData();
  99. void testFromHex();
  100. void testParsePrioritizePieceRange();
  101. void testApplyDir();
  102. void testFixTaintedBasename();
  103. };
  104. CPPUNIT_TEST_SUITE_REGISTRATION( UtilTest );
  105. void UtilTest::testTrim() {
  106. std::string str1 = "aria2";
  107. CPPUNIT_ASSERT_EQUAL(str1, util::trim("aria2"));
  108. CPPUNIT_ASSERT_EQUAL(str1, util::trim(" aria2"));
  109. CPPUNIT_ASSERT_EQUAL(str1, util::trim(" aria2 "));
  110. CPPUNIT_ASSERT_EQUAL(str1, util::trim(" aria2 "));
  111. std::string str2 = "aria2 debut";
  112. CPPUNIT_ASSERT_EQUAL(str2, util::trim("aria2 debut"));
  113. CPPUNIT_ASSERT_EQUAL(str2, util::trim(" aria2 debut "));
  114. std::string str3 = "";
  115. CPPUNIT_ASSERT_EQUAL(str3, util::trim(""));
  116. CPPUNIT_ASSERT_EQUAL(str3, util::trim(" "));
  117. CPPUNIT_ASSERT_EQUAL(str3, util::trim(" "));
  118. std::string str4 = "A";
  119. CPPUNIT_ASSERT_EQUAL(str4, util::trim("A"));
  120. CPPUNIT_ASSERT_EQUAL(str4, util::trim(" A "));
  121. CPPUNIT_ASSERT_EQUAL(str4, util::trim(" A "));
  122. }
  123. void UtilTest::testSplit() {
  124. std::pair<std::string, std::string> p1;
  125. util::split(p1, "name=value", '=');
  126. CPPUNIT_ASSERT_EQUAL(std::string("name"), p1.first);
  127. CPPUNIT_ASSERT_EQUAL(std::string("value"), p1.second);
  128. util::split(p1, " name = value ", '=');
  129. CPPUNIT_ASSERT_EQUAL(std::string("name"), p1.first);
  130. CPPUNIT_ASSERT_EQUAL(std::string("value"), p1.second);
  131. util::split(p1, "=value", '=');
  132. CPPUNIT_ASSERT_EQUAL(std::string(""), p1.first);
  133. CPPUNIT_ASSERT_EQUAL(std::string("value"), p1.second);
  134. util::split(p1, "name=", '=');
  135. CPPUNIT_ASSERT_EQUAL(std::string("name"), p1.first);
  136. CPPUNIT_ASSERT_EQUAL(std::string(""), p1.second);
  137. util::split(p1, "name", '=');
  138. CPPUNIT_ASSERT_EQUAL(std::string("name"), p1.first);
  139. CPPUNIT_ASSERT_EQUAL(std::string(""), p1.second);
  140. }
  141. void UtilTest::testSplit_many() {
  142. std::deque<std::string> v1;
  143. util::split("name1=value1; name2=value2; name3=value3",std::back_inserter(v1),
  144. ";", true);
  145. CPPUNIT_ASSERT_EQUAL(3, (int)v1.size());
  146. std::deque<std::string>::iterator itr = v1.begin();
  147. CPPUNIT_ASSERT_EQUAL(std::string("name1=value1"), *itr++);
  148. CPPUNIT_ASSERT_EQUAL(std::string("name2=value2"), *itr++);
  149. CPPUNIT_ASSERT_EQUAL(std::string("name3=value3"), *itr++);
  150. v1.clear();
  151. util::split("name1=value1; name2=value2; name3=value3",std::back_inserter(v1),
  152. ";", false);
  153. CPPUNIT_ASSERT_EQUAL(3, (int)v1.size());
  154. itr = v1.begin();
  155. CPPUNIT_ASSERT_EQUAL(std::string("name1=value1"), *itr++);
  156. CPPUNIT_ASSERT_EQUAL(std::string(" name2=value2"), *itr++);
  157. CPPUNIT_ASSERT_EQUAL(std::string(" name3=value3"), *itr++);
  158. }
  159. void UtilTest::testEndsWith() {
  160. std::string target = "abcdefg";
  161. std::string part = "fg";
  162. CPPUNIT_ASSERT(util::endsWith(target, part));
  163. target = "abdefg";
  164. part = "g";
  165. CPPUNIT_ASSERT(util::endsWith(target, part));
  166. target = "abdefg";
  167. part = "eg";
  168. CPPUNIT_ASSERT(!util::endsWith(target, part));
  169. target = "g";
  170. part = "eg";
  171. CPPUNIT_ASSERT(!util::endsWith(target, part));
  172. target = "g";
  173. part = "g";
  174. CPPUNIT_ASSERT(util::endsWith(target, part));
  175. target = "g";
  176. part = "";
  177. CPPUNIT_ASSERT(util::endsWith(target, part));
  178. target = "";
  179. part = "";
  180. CPPUNIT_ASSERT(util::endsWith(target, part));
  181. target = "";
  182. part = "g";
  183. CPPUNIT_ASSERT(!util::endsWith(target, part));
  184. }
  185. void UtilTest::testReplace() {
  186. CPPUNIT_ASSERT_EQUAL(std::string("abc\n"), util::replace("abc\r\n", "\r", ""));
  187. CPPUNIT_ASSERT_EQUAL(std::string("abc"), util::replace("abc\r\n", "\r\n", ""));
  188. CPPUNIT_ASSERT_EQUAL(std::string(""), util::replace("", "\r\n", ""));
  189. CPPUNIT_ASSERT_EQUAL(std::string("abc"), util::replace("abc", "", "a"));
  190. CPPUNIT_ASSERT_EQUAL(std::string("xbc"), util::replace("abc", "a", "x"));
  191. }
  192. void UtilTest::testStartsWith() {
  193. std::string target;
  194. std::string part;
  195. target = "abcdefg";
  196. part = "abc";
  197. CPPUNIT_ASSERT(util::startsWith(target, part));
  198. target = "abcdefg";
  199. part = "abx";
  200. CPPUNIT_ASSERT(!util::startsWith(target, part));
  201. target = "abcdefg";
  202. part = "bcd";
  203. CPPUNIT_ASSERT(!util::startsWith(target, part));
  204. target = "";
  205. part = "a";
  206. CPPUNIT_ASSERT(!util::startsWith(target, part));
  207. target = "";
  208. part = "";
  209. CPPUNIT_ASSERT(util::startsWith(target, part));
  210. target = "a";
  211. part = "";
  212. CPPUNIT_ASSERT(util::startsWith(target, part));
  213. target = "a";
  214. part = "a";
  215. CPPUNIT_ASSERT(util::startsWith(target, part));
  216. }
  217. void UtilTest::testGetContentDispositionFilename() {
  218. std::string h1 = "attachment; filename=\"aria2.tar.bz2\"";
  219. CPPUNIT_ASSERT_EQUAL(std::string("aria2.tar.bz2"), util::getContentDispositionFilename(h1));
  220. std::string h2 = "attachment; filename=\"\"";
  221. CPPUNIT_ASSERT_EQUAL(std::string(""), util::getContentDispositionFilename(h2));
  222. std::string h3 = "attachment; filename=\"";
  223. CPPUNIT_ASSERT_EQUAL(std::string(""), util::getContentDispositionFilename(h3));
  224. std::string h4 = "attachment;";
  225. CPPUNIT_ASSERT_EQUAL(std::string(""), util::getContentDispositionFilename(h4));
  226. std::string h5 = "attachment; filename=aria2.tar.bz2";
  227. CPPUNIT_ASSERT_EQUAL(std::string("aria2.tar.bz2"), util::getContentDispositionFilename(h5));
  228. std::string h6 = "attachment; filename='aria2.tar.bz2'";
  229. CPPUNIT_ASSERT_EQUAL(std::string("aria2.tar.bz2"), util::getContentDispositionFilename(h6));
  230. std::string h7 = "attachment; filename='aria2.tar.bz2";
  231. CPPUNIT_ASSERT_EQUAL(std::string("aria2.tar.bz2"), util::getContentDispositionFilename(h7));
  232. std::string h8 = "attachment; filename=aria2.tar.bz2; creation-date=20 Jun 2007 00:00:00 GMT";
  233. CPPUNIT_ASSERT_EQUAL(std::string("aria2.tar.bz2"), util::getContentDispositionFilename(h8));
  234. std::string h9 = "attachment; filename=\"aria2.tar.bz2; creation-date=20 Jun 2007 00:00:00 GMT\"";
  235. CPPUNIT_ASSERT_EQUAL(std::string("aria2.tar.bz2; creation-date=20 Jun 2007 00:00:00 GMT"), util::getContentDispositionFilename(h9));
  236. std::string h10 = "attachment; filename=";
  237. CPPUNIT_ASSERT_EQUAL(std::string(""), util::getContentDispositionFilename(h10));
  238. std::string h11 = "attachment; filename=;";
  239. CPPUNIT_ASSERT_EQUAL(std::string(""), util::getContentDispositionFilename(h11));
  240. std::string filenameWithDir = "attachment; filename=dir/file";
  241. CPPUNIT_ASSERT_EQUAL(std::string("file"),
  242. util::getContentDispositionFilename(filenameWithDir));
  243. std::string parentDir = "attachment; filename=..";
  244. CPPUNIT_ASSERT_EQUAL(std::string(),
  245. util::getContentDispositionFilename(parentDir));
  246. std::string currentDir = "attachment; filename=.";
  247. CPPUNIT_ASSERT_EQUAL(std::string(),
  248. util::getContentDispositionFilename(currentDir));
  249. }
  250. class Printer {
  251. public:
  252. template<class T>
  253. void operator()(T t) {
  254. std::cerr << t << ", ";
  255. }
  256. };
  257. void UtilTest::testRandomAlpha() {
  258. SharedHandle<Randomizer> rand(new FixedNumberRandomizer());
  259. std::string s = util::randomAlpha(8, rand);
  260. CPPUNIT_ASSERT_EQUAL(std::string("AAAAAAAA"), s);
  261. }
  262. void UtilTest::testToUpper() {
  263. std::string src = "608cabc0f2fa18c260cafd974516865c772363d5";
  264. std::string upp = "608CABC0F2FA18C260CAFD974516865C772363D5";
  265. CPPUNIT_ASSERT_EQUAL(upp, util::toUpper(src));
  266. }
  267. void UtilTest::testToLower() {
  268. std::string src = "608CABC0F2FA18C260CAFD974516865C772363D5";
  269. std::string upp = "608cabc0f2fa18c260cafd974516865c772363d5";
  270. CPPUNIT_ASSERT_EQUAL(upp, util::toLower(src));
  271. }
  272. void UtilTest::testUrldecode() {
  273. std::string src = "http://aria2.sourceforge.net/aria2%200.7.0%20docs.html";
  274. CPPUNIT_ASSERT_EQUAL(std::string("http://aria2.sourceforge.net/aria2 0.7.0 docs.html"),
  275. util::urldecode(src));
  276. std::string src2 = "aria2+aria2";
  277. CPPUNIT_ASSERT_EQUAL(std::string("aria2+aria2"), util::urldecode(src2));
  278. std::string src3 = "%5t%20";
  279. CPPUNIT_ASSERT_EQUAL(std::string("%5t "), util::urldecode(src3));
  280. std::string src4 = "%";
  281. CPPUNIT_ASSERT_EQUAL(std::string("%"), util::urldecode(src4));
  282. std::string src5 = "%3";
  283. CPPUNIT_ASSERT_EQUAL(std::string("%3"), util::urldecode(src5));
  284. std::string src6 = "%2f";
  285. CPPUNIT_ASSERT_EQUAL(std::string("/"), util::urldecode(src6));
  286. }
  287. void UtilTest::testGetRealSize()
  288. {
  289. CPPUNIT_ASSERT_EQUAL((int64_t)4294967296LL, util::getRealSize("4096M"));
  290. CPPUNIT_ASSERT_EQUAL((int64_t)1024, util::getRealSize("1K"));
  291. try {
  292. util::getRealSize("");
  293. CPPUNIT_FAIL("exception must be thrown.");
  294. } catch(Exception& e) {
  295. std::cerr << e.stackTrace();
  296. }
  297. try {
  298. util::getRealSize("foo");
  299. CPPUNIT_FAIL("exception must be thrown.");
  300. } catch(Exception& e) {
  301. std::cerr << e.stackTrace();
  302. }
  303. try {
  304. util::getRealSize("-1");
  305. CPPUNIT_FAIL("exception must be thrown.");
  306. } catch(Exception& e) {
  307. std::cerr << e.stackTrace();
  308. }
  309. try {
  310. util::getRealSize("9223372036854775807K");
  311. CPPUNIT_FAIL("exception must be thrown.");
  312. } catch(Exception& e) {
  313. std::cerr << e.stackTrace();
  314. }
  315. try {
  316. util::getRealSize("9223372036854775807M");
  317. CPPUNIT_FAIL("exception must be thrown.");
  318. } catch(Exception& e) {
  319. std::cerr << e.stackTrace();
  320. }
  321. }
  322. void UtilTest::testAbbrevSize()
  323. {
  324. CPPUNIT_ASSERT_EQUAL(std::string("4,096.0Mi"), util::abbrevSize(4294967296LL));
  325. CPPUNIT_ASSERT_EQUAL(std::string("1.0Ki"), util::abbrevSize(1024));
  326. CPPUNIT_ASSERT_EQUAL(std::string("1,023"), util::abbrevSize(1023));
  327. CPPUNIT_ASSERT_EQUAL(std::string("0"), util::abbrevSize(0));
  328. CPPUNIT_ASSERT_EQUAL(std::string("1.1Ki"), util::abbrevSize(1127));
  329. CPPUNIT_ASSERT_EQUAL(std::string("1.5Mi"), util::abbrevSize(1572864));
  330. }
  331. void UtilTest::testToStream()
  332. {
  333. std::ostringstream os;
  334. SharedHandle<FileEntry> f1(new FileEntry("aria2.tar.bz2", 12300, 0));
  335. SharedHandle<FileEntry> f2(new FileEntry("aria2.txt", 556, 0));
  336. std::deque<SharedHandle<FileEntry> > entries;
  337. entries.push_back(f1);
  338. entries.push_back(f2);
  339. util::toStream(entries.begin(), entries.end(), os);
  340. CPPUNIT_ASSERT_EQUAL(
  341. std::string("Files:\n"
  342. "idx|path/length\n"
  343. "===+===========================================================================\n"
  344. " 1|aria2.tar.bz2\n"
  345. " |12.0KiB (12,300)\n"
  346. "---+---------------------------------------------------------------------------\n"
  347. " 2|aria2.txt\n"
  348. " |556B (556)\n"
  349. "---+---------------------------------------------------------------------------\n"),
  350. os.str());
  351. }
  352. void UtilTest::testIsNumber()
  353. {
  354. CPPUNIT_ASSERT_EQUAL(true, util::isNumber("000"));
  355. CPPUNIT_ASSERT_EQUAL(false, util::isNumber("a"));
  356. CPPUNIT_ASSERT_EQUAL(false, util::isNumber("0a"));
  357. CPPUNIT_ASSERT_EQUAL(false, util::isNumber(""));
  358. CPPUNIT_ASSERT_EQUAL(false, util::isNumber(" "));
  359. }
  360. void UtilTest::testIsLowercase()
  361. {
  362. CPPUNIT_ASSERT_EQUAL(true, util::isLowercase("alpha"));
  363. CPPUNIT_ASSERT_EQUAL(false, util::isLowercase("Alpha"));
  364. CPPUNIT_ASSERT_EQUAL(false, util::isLowercase("1alpha"));
  365. CPPUNIT_ASSERT_EQUAL(false, util::isLowercase(""));
  366. CPPUNIT_ASSERT_EQUAL(false, util::isLowercase(" "));
  367. }
  368. void UtilTest::testIsUppercase()
  369. {
  370. CPPUNIT_ASSERT_EQUAL(true, util::isUppercase("ALPHA"));
  371. CPPUNIT_ASSERT_EQUAL(false, util::isUppercase("Alpha"));
  372. CPPUNIT_ASSERT_EQUAL(false, util::isUppercase("1ALPHA"));
  373. CPPUNIT_ASSERT_EQUAL(false, util::isUppercase(""));
  374. CPPUNIT_ASSERT_EQUAL(false, util::isUppercase(" "));
  375. }
  376. void UtilTest::testAlphaToNum()
  377. {
  378. CPPUNIT_ASSERT_EQUAL(0U, util::alphaToNum("a"));
  379. CPPUNIT_ASSERT_EQUAL(0U, util::alphaToNum("aa"));
  380. CPPUNIT_ASSERT_EQUAL(1U, util::alphaToNum("b"));
  381. CPPUNIT_ASSERT_EQUAL(675U, util::alphaToNum("zz")); // 25*26+25
  382. CPPUNIT_ASSERT_EQUAL(675U, util::alphaToNum("ZZ")); // 25*26+25
  383. CPPUNIT_ASSERT_EQUAL(0U, util::alphaToNum(""));
  384. CPPUNIT_ASSERT_EQUAL(4294967295U, util::alphaToNum("NXMRLXV"));
  385. CPPUNIT_ASSERT_EQUAL(0U, util::alphaToNum("NXMRLXW")); // uint32_t overflow
  386. }
  387. void UtilTest::testMkdirs()
  388. {
  389. std::string dir = "/tmp/aria2-UtilTest-testMkdirs";
  390. File d(dir);
  391. if(d.exists()) {
  392. CPPUNIT_ASSERT(d.remove());
  393. }
  394. CPPUNIT_ASSERT(!d.exists());
  395. util::mkdirs(dir);
  396. CPPUNIT_ASSERT(d.isDir());
  397. std::string file = "./UtilTest.cc";
  398. File f(file);
  399. CPPUNIT_ASSERT(f.isFile());
  400. try {
  401. util::mkdirs(file);
  402. CPPUNIT_FAIL("exception must be thrown.");
  403. } catch(DlAbortEx& ex) {
  404. std::cerr << ex.stackTrace() << std::endl;
  405. }
  406. }
  407. void UtilTest::testConvertBitfield()
  408. {
  409. BitfieldMan srcBitfield(384*1024, 256*1024*256+1);
  410. BitfieldMan destBitfield(512*1024, srcBitfield.getTotalLength());
  411. srcBitfield.setAllBit();
  412. srcBitfield.unsetBit(2);// <- range [768, 1152)
  413. // which corresponds to the index [1,2] in destBitfield
  414. util::convertBitfield(&destBitfield, &srcBitfield);
  415. CPPUNIT_ASSERT_EQUAL(std::string("9fffffffffffffffffffffffffffffff80"),
  416. util::toHex(destBitfield.getBitfield(),
  417. destBitfield.getBitfieldLength()));
  418. }
  419. void UtilTest::testParseIntRange()
  420. {
  421. IntSequence seq = util::parseIntRange("1,3-8,10");
  422. CPPUNIT_ASSERT(seq.hasNext());
  423. CPPUNIT_ASSERT_EQUAL((int32_t)1, seq.next());
  424. CPPUNIT_ASSERT(seq.hasNext());
  425. CPPUNIT_ASSERT_EQUAL((int32_t)3, seq.next());
  426. CPPUNIT_ASSERT(seq.hasNext());
  427. CPPUNIT_ASSERT_EQUAL((int32_t)4, seq.next());
  428. CPPUNIT_ASSERT(seq.hasNext());
  429. CPPUNIT_ASSERT_EQUAL((int32_t)5, seq.next());
  430. CPPUNIT_ASSERT(seq.hasNext());
  431. CPPUNIT_ASSERT_EQUAL((int32_t)6, seq.next());
  432. CPPUNIT_ASSERT(seq.hasNext());
  433. CPPUNIT_ASSERT_EQUAL((int32_t)7, seq.next());
  434. CPPUNIT_ASSERT(seq.hasNext());
  435. CPPUNIT_ASSERT_EQUAL((int32_t)8, seq.next());
  436. CPPUNIT_ASSERT(seq.hasNext());
  437. CPPUNIT_ASSERT_EQUAL((int32_t)10, seq.next());
  438. CPPUNIT_ASSERT(!seq.hasNext());
  439. CPPUNIT_ASSERT_EQUAL((int32_t)0, seq.next());
  440. }
  441. void UtilTest::testParseIntRange_invalidRange()
  442. {
  443. try {
  444. IntSequence seq = util::parseIntRange("-1");
  445. CPPUNIT_FAIL("exception must be thrown.");
  446. } catch(Exception& e) {
  447. std::cerr << e.stackTrace();
  448. }
  449. try {
  450. IntSequence seq = util::parseIntRange("2147483648");
  451. CPPUNIT_FAIL("exception must be thrown.");
  452. } catch(Exception& e) {
  453. std::cerr << e.stackTrace();
  454. }
  455. try {
  456. IntSequence seq = util::parseIntRange("2147483647-2147483648");
  457. CPPUNIT_FAIL("exception must be thrown.");
  458. } catch(Exception& e) {
  459. std::cerr << e.stackTrace();
  460. }
  461. try {
  462. IntSequence seq = util::parseIntRange("1-2x");
  463. CPPUNIT_FAIL("exception must be thrown.");
  464. } catch(Exception& e) {
  465. std::cerr << e.stackTrace();
  466. }
  467. try {
  468. IntSequence seq = util::parseIntRange("3x-4");
  469. CPPUNIT_FAIL("exception must be thrown.");
  470. } catch(Exception& e) {
  471. std::cerr << e.stackTrace();
  472. }
  473. }
  474. void UtilTest::testParseInt()
  475. {
  476. CPPUNIT_ASSERT_EQUAL(-1, util::parseInt(" -1 "));
  477. CPPUNIT_ASSERT_EQUAL(2147483647, util::parseInt("2147483647"));
  478. try {
  479. util::parseInt("2147483648");
  480. CPPUNIT_FAIL("exception must be thrown.");
  481. } catch(Exception& e) {
  482. std::cerr << e.stackTrace();
  483. }
  484. try {
  485. util::parseInt("-2147483649");
  486. CPPUNIT_FAIL("exception must be thrown.");
  487. } catch(Exception& e) {
  488. std::cerr << e.stackTrace();
  489. }
  490. try {
  491. util::parseInt("12x");
  492. CPPUNIT_FAIL("exception must be thrown.");
  493. } catch(Exception& e) {
  494. std::cerr << e.stackTrace();
  495. }
  496. try {
  497. util::parseInt("");
  498. CPPUNIT_FAIL("exception must be thrown.");
  499. } catch(Exception& e) {
  500. std::cerr << e.stackTrace();
  501. }
  502. }
  503. void UtilTest::testParseUInt()
  504. {
  505. CPPUNIT_ASSERT_EQUAL(4294967295U, util::parseUInt(" 4294967295 "));
  506. try {
  507. util::parseUInt("-1");
  508. CPPUNIT_FAIL("exception must be thrown.");
  509. } catch(Exception& e) {
  510. std::cerr << e.stackTrace();
  511. }
  512. try {
  513. util::parseUInt("4294967296");
  514. CPPUNIT_FAIL("exception must be thrown.");
  515. } catch(Exception& e) {
  516. std::cerr << e.stackTrace();
  517. }
  518. }
  519. void UtilTest::testParseLLInt()
  520. {
  521. CPPUNIT_ASSERT_EQUAL((int64_t)-1LL, util::parseLLInt(" -1 "));
  522. CPPUNIT_ASSERT_EQUAL((int64_t)9223372036854775807LL,
  523. util::parseLLInt("9223372036854775807"));
  524. try {
  525. util::parseLLInt("9223372036854775808");
  526. CPPUNIT_FAIL("exception must be thrown.");
  527. } catch(Exception& e) {
  528. std::cerr << e.stackTrace();
  529. }
  530. try {
  531. util::parseLLInt("-9223372036854775809");
  532. CPPUNIT_FAIL("exception must be thrown.");
  533. } catch(Exception& e) {
  534. std::cerr << e.stackTrace();
  535. }
  536. try {
  537. util::parseLLInt("12x");
  538. CPPUNIT_FAIL("exception must be thrown.");
  539. } catch(Exception& e) {
  540. std::cerr << e.stackTrace();
  541. }
  542. try {
  543. util::parseLLInt("");
  544. CPPUNIT_FAIL("exception must be thrown.");
  545. } catch(Exception& e) {
  546. std::cerr << e.stackTrace();
  547. }
  548. }
  549. void UtilTest::testParseULLInt()
  550. {
  551. CPPUNIT_ASSERT_EQUAL((uint64_t)18446744073709551615ULL,
  552. util::parseULLInt("18446744073709551615"));
  553. try {
  554. util::parseUInt("-1");
  555. CPPUNIT_FAIL("exception must be thrown.");
  556. } catch(Exception& e) {
  557. std::cerr << e.stackTrace();
  558. }
  559. try {
  560. util::parseLLInt("18446744073709551616");
  561. CPPUNIT_FAIL("exception must be thrown.");
  562. } catch(Exception& e) {
  563. std::cerr << e.stackTrace();
  564. }
  565. }
  566. void UtilTest::testToString_binaryStream()
  567. {
  568. SharedHandle<DiskWriter> dw(new ByteArrayDiskWriter());
  569. std::string data(16*1024+256, 'a');
  570. dw->initAndOpenFile();
  571. dw->writeData((const unsigned char*)data.c_str(), data.size(), 0);
  572. std::string readData = util::toString(dw);
  573. CPPUNIT_ASSERT_EQUAL(data, readData);
  574. }
  575. void UtilTest::testItos()
  576. {
  577. {
  578. int i = 0;
  579. CPPUNIT_ASSERT_EQUAL(std::string("0"), util::itos(i));
  580. }
  581. {
  582. int i = 100;
  583. CPPUNIT_ASSERT_EQUAL(std::string("100"), util::itos(i, true));
  584. }
  585. {
  586. int i = 100;
  587. CPPUNIT_ASSERT_EQUAL(std::string("100"), util::itos(i));
  588. }
  589. {
  590. int i = 12345;
  591. CPPUNIT_ASSERT_EQUAL(std::string("12,345"), util::itos(i, true));
  592. }
  593. {
  594. int i = 12345;
  595. CPPUNIT_ASSERT_EQUAL(std::string("12345"), util::itos(i));
  596. }
  597. {
  598. int i = -12345;
  599. CPPUNIT_ASSERT_EQUAL(std::string("-12,345"), util::itos(i, true));
  600. }
  601. {
  602. int64_t i = INT64_MAX;
  603. CPPUNIT_ASSERT_EQUAL(std::string("9,223,372,036,854,775,807"), util::itos(i, true));
  604. }
  605. }
  606. void UtilTest::testUitos()
  607. {
  608. {
  609. uint16_t i = 12345;
  610. CPPUNIT_ASSERT_EQUAL(std::string("12345"), util::uitos(i));
  611. }
  612. {
  613. int16_t i = -12345;
  614. CPPUNIT_ASSERT_EQUAL(std::string("/.-,+"), util::uitos(i));
  615. }
  616. }
  617. void UtilTest::testNtoh64()
  618. {
  619. uint64_t x = 0xff00ff00ee00ee00LL;
  620. #ifdef WORDS_BIGENDIAN
  621. CPPUNIT_ASSERT_EQUAL(x, ntoh64(x));
  622. CPPUNIT_ASSERT_EQUAL(x, hton64(x));
  623. #else // !WORDS_BIGENDIAN
  624. uint64_t y = 0x00ee00ee00ff00ffLL;
  625. CPPUNIT_ASSERT_EQUAL(y, ntoh64(x));
  626. CPPUNIT_ASSERT_EQUAL(x, hton64(y));
  627. #endif // !WORDS_BIGENDIAN
  628. }
  629. void UtilTest::testUrlencode()
  630. {
  631. CPPUNIT_ASSERT_EQUAL
  632. (std::string("%3A%2F%3F%23%5B%5D%40%21%25%26%27%28%29%2A%2B%2C%3B%3D"),
  633. util::urlencode(":/?#[]@!%&'()*+,;="));
  634. std::string unreserved =
  635. "ABCDEFGHIJKLMNOPQRSTUVWXYZ"
  636. "abcdefghijklmnopqrstuvwxyz"
  637. "0123456789"
  638. "-._~";
  639. CPPUNIT_ASSERT_EQUAL(unreserved, util::urlencode(unreserved));
  640. CPPUNIT_ASSERT_EQUAL(std::string("1%5EA%20"), util::urlencode("1^A "));
  641. }
  642. void UtilTest::testHtmlEscape()
  643. {
  644. CPPUNIT_ASSERT_EQUAL(std::string("aria2&lt;&gt;&quot;&#39;util"),
  645. util::htmlEscape("aria2<>\"'util"));
  646. }
  647. void UtilTest::testJoinPath()
  648. {
  649. const std::string dir1dir2file[] = { "dir1", "dir2", "file" };
  650. CPPUNIT_ASSERT_EQUAL
  651. (std::string("dir1/dir2/file"),
  652. util::joinPath(&dir1dir2file[0],
  653. &dir1dir2file[arrayLength(dir1dir2file)]));
  654. const std::string dirparentfile[] = { "dir", "..", "file" };
  655. CPPUNIT_ASSERT_EQUAL
  656. (std::string("file"),
  657. util::joinPath(&dirparentfile[0],
  658. &dirparentfile[arrayLength(dirparentfile)]));
  659. const std::string dirparentparentfile[] = { "dir", "..", "..", "file" };
  660. CPPUNIT_ASSERT_EQUAL
  661. (std::string("file"),
  662. util::joinPath(&dirparentparentfile[0],
  663. &dirparentparentfile[arrayLength(dirparentparentfile)]));
  664. const std::string dirdotfile[] = { "dir", ".", "file" };
  665. CPPUNIT_ASSERT_EQUAL(std::string("dir/file"),
  666. util::joinPath(&dirdotfile[0],
  667. &dirdotfile[arrayLength(dirdotfile)]));
  668. const std::string empty[] = {};
  669. CPPUNIT_ASSERT_EQUAL(std::string(""), util::joinPath(&empty[0], &empty[0]));
  670. const std::string parentdot[] = { "..", "." };
  671. CPPUNIT_ASSERT_EQUAL(std::string(""),
  672. util::joinPath(&parentdot[0],
  673. &parentdot[arrayLength(parentdot)]));
  674. }
  675. void UtilTest::testParseIndexPath()
  676. {
  677. std::map<size_t, std::string>::value_type p = util::parseIndexPath("1=foo");
  678. CPPUNIT_ASSERT_EQUAL((size_t)1, p.first);
  679. CPPUNIT_ASSERT_EQUAL(std::string("foo"), p.second);
  680. try {
  681. util::parseIndexPath("1X=foo");
  682. CPPUNIT_FAIL("exception must be thrown.");
  683. } catch(Exception& e) {
  684. // success
  685. }
  686. try {
  687. util::parseIndexPath("1=");
  688. CPPUNIT_FAIL("exception must be thrown.");
  689. } catch(Exception& e) {
  690. // success
  691. }
  692. }
  693. void UtilTest::testCreateIndexPathMap()
  694. {
  695. std::stringstream in
  696. ("1=/tmp/myfile\n"
  697. "100=/myhome/mypicture.png\n");
  698. std::map<size_t, std::string> m = util::createIndexPathMap(in);
  699. CPPUNIT_ASSERT_EQUAL((size_t)2, m.size());
  700. CPPUNIT_ASSERT(m.find(1) != m.end());
  701. CPPUNIT_ASSERT_EQUAL(std::string("/tmp/myfile"), m[1]);
  702. CPPUNIT_ASSERT(m.find(100) != m.end());
  703. CPPUNIT_ASSERT_EQUAL(std::string("/myhome/mypicture.png"), m[100]);
  704. }
  705. void UtilTest::testGenerateRandomData()
  706. {
  707. unsigned char data1[20];
  708. util::generateRandomData(data1, sizeof(data1));
  709. unsigned char data2[20];
  710. util::generateRandomData(data2, sizeof(data2));
  711. CPPUNIT_ASSERT(memcmp(data1, data2, sizeof(data1)) != 0);
  712. }
  713. void UtilTest::testFromHex()
  714. {
  715. std::string src;
  716. std::string dest;
  717. src = "0011fF";
  718. dest = util::fromHex(src);
  719. CPPUNIT_ASSERT_EQUAL((size_t)3, dest.size());
  720. CPPUNIT_ASSERT_EQUAL((char)0x00, dest[0]);
  721. CPPUNIT_ASSERT_EQUAL((char)0x11, dest[1]);
  722. CPPUNIT_ASSERT_EQUAL((char)0xff, dest[2]);
  723. src = "0011f";
  724. CPPUNIT_ASSERT(util::fromHex(src).empty());
  725. src = "001g";
  726. CPPUNIT_ASSERT(util::fromHex(src).empty());
  727. }
  728. void UtilTest::testParsePrioritizePieceRange()
  729. {
  730. // piece index
  731. // 0 1 2 3 4 5 6 7
  732. // | | | |
  733. // file1 | | |
  734. // | | |
  735. // file2 | |
  736. // file3 |
  737. // | |
  738. // file4 |
  739. size_t pieceLength = 1024;
  740. std::vector<SharedHandle<FileEntry> > entries(4, SharedHandle<FileEntry>());
  741. entries[0].reset(new FileEntry("file1", 1024, 0));
  742. entries[1].reset(new FileEntry("file2",2560,entries[0]->getLastOffset()));
  743. entries[2].reset(new FileEntry("file3",0,entries[1]->getLastOffset()));
  744. entries[3].reset(new FileEntry("file4",3584,entries[2]->getLastOffset()));
  745. std::vector<size_t> result;
  746. util::parsePrioritizePieceRange(result, "head=1", entries, pieceLength);
  747. CPPUNIT_ASSERT_EQUAL((size_t)3, result.size());
  748. CPPUNIT_ASSERT_EQUAL((size_t)0, result[0]);
  749. CPPUNIT_ASSERT_EQUAL((size_t)1, result[1]);
  750. CPPUNIT_ASSERT_EQUAL((size_t)3, result[2]);
  751. result.clear();
  752. util::parsePrioritizePieceRange(result, "tail=1", entries, pieceLength);
  753. CPPUNIT_ASSERT_EQUAL((size_t)3, result.size());
  754. CPPUNIT_ASSERT_EQUAL((size_t)0, result[0]);
  755. CPPUNIT_ASSERT_EQUAL((size_t)3, result[1]);
  756. CPPUNIT_ASSERT_EQUAL((size_t)6, result[2]);
  757. result.clear();
  758. util::parsePrioritizePieceRange(result, "head=1K", entries, pieceLength);
  759. CPPUNIT_ASSERT_EQUAL((size_t)4, result.size());
  760. CPPUNIT_ASSERT_EQUAL((size_t)0, result[0]);
  761. CPPUNIT_ASSERT_EQUAL((size_t)1, result[1]);
  762. CPPUNIT_ASSERT_EQUAL((size_t)3, result[2]);
  763. CPPUNIT_ASSERT_EQUAL((size_t)4, result[3]);
  764. result.clear();
  765. util::parsePrioritizePieceRange(result, "head", entries, pieceLength, 1024);
  766. CPPUNIT_ASSERT_EQUAL((size_t)4, result.size());
  767. CPPUNIT_ASSERT_EQUAL((size_t)0, result[0]);
  768. CPPUNIT_ASSERT_EQUAL((size_t)1, result[1]);
  769. CPPUNIT_ASSERT_EQUAL((size_t)3, result[2]);
  770. CPPUNIT_ASSERT_EQUAL((size_t)4, result[3]);
  771. result.clear();
  772. util::parsePrioritizePieceRange(result, "tail=1K", entries, pieceLength);
  773. CPPUNIT_ASSERT_EQUAL((size_t)4, result.size());
  774. CPPUNIT_ASSERT_EQUAL((size_t)0, result[0]);
  775. CPPUNIT_ASSERT_EQUAL((size_t)2, result[1]);
  776. CPPUNIT_ASSERT_EQUAL((size_t)3, result[2]);
  777. CPPUNIT_ASSERT_EQUAL((size_t)6, result[3]);
  778. result.clear();
  779. util::parsePrioritizePieceRange(result, "tail", entries, pieceLength, 1024);
  780. CPPUNIT_ASSERT_EQUAL((size_t)4, result.size());
  781. CPPUNIT_ASSERT_EQUAL((size_t)0, result[0]);
  782. CPPUNIT_ASSERT_EQUAL((size_t)2, result[1]);
  783. CPPUNIT_ASSERT_EQUAL((size_t)3, result[2]);
  784. CPPUNIT_ASSERT_EQUAL((size_t)6, result[3]);
  785. result.clear();
  786. util::parsePrioritizePieceRange(result, "head=1,tail=1", entries, pieceLength);
  787. CPPUNIT_ASSERT_EQUAL((size_t)4, result.size());
  788. CPPUNIT_ASSERT_EQUAL((size_t)0, result[0]);
  789. CPPUNIT_ASSERT_EQUAL((size_t)1, result[1]);
  790. CPPUNIT_ASSERT_EQUAL((size_t)3, result[2]);
  791. CPPUNIT_ASSERT_EQUAL((size_t)6, result[3]);
  792. result.clear();
  793. util::parsePrioritizePieceRange(result, "head=300M,tail=300M",
  794. entries, pieceLength);
  795. CPPUNIT_ASSERT_EQUAL((size_t)7, result.size());
  796. for(size_t i = 0; i < 7; ++i) {
  797. CPPUNIT_ASSERT_EQUAL(i, result[i]);
  798. }
  799. result.clear();
  800. util::parsePrioritizePieceRange(result, "", entries, pieceLength);
  801. CPPUNIT_ASSERT(result.empty());
  802. }
  803. void UtilTest::testApplyDir()
  804. {
  805. CPPUNIT_ASSERT_EQUAL(std::string("./pred"), util::applyDir("", "pred"));
  806. CPPUNIT_ASSERT_EQUAL(std::string("/pred"), util::applyDir("/", "pred"));
  807. CPPUNIT_ASSERT_EQUAL(std::string("./pred"), util::applyDir(".", "pred"));
  808. CPPUNIT_ASSERT_EQUAL(std::string("/dl/pred"), util::applyDir("/dl", "pred"));
  809. }
  810. void UtilTest::testFixTaintedBasename()
  811. {
  812. CPPUNIT_ASSERT_EQUAL(std::string("a_b"), util::fixTaintedBasename("a/b"));
  813. CPPUNIT_ASSERT_EQUAL(std::string("a_b"), util::fixTaintedBasename("a\\b"));
  814. CPPUNIT_ASSERT_EQUAL(std::string("a__b"), util::fixTaintedBasename("a\\/b"));
  815. }
  816. } // namespace aria2